From 115f00ff1a6e00f53a2753252530f78ccdfafa3f Mon Sep 17 00:00:00 2001 From: ganfra Date: Thu, 22 Jul 2021 10:12:46 +0200 Subject: [PATCH 1/4] Start experiment on log tags --- .../android/sdk/api/logger/LoggerTag.kt | 32 ++++++++ .../session/call/CallEventProcessor.kt | 11 +-- .../session/call/CallSignalingHandler.kt | 23 ++++-- .../vector/app/core/services/CallService.kt | 23 +++--- .../app/features/call/webrtc/WebRtcCall.kt | 81 ++++++++++--------- .../features/call/webrtc/WebRtcCallManager.kt | 53 ++++++------ 6 files changed, 135 insertions(+), 88 deletions(-) create mode 100644 matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt new file mode 100644 index 0000000000..074bd5cf6c --- /dev/null +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt @@ -0,0 +1,32 @@ +/* + * Copyright (c) 2021 The Matrix.org Foundation C.I.C. + * + * 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.matrix.android.sdk.api.logger + +/** + * Some tags used on Timber. + */ +open class LoggerTag(private val value: String, private val parentTag: LoggerTag? = null) { + + object VOIP : LoggerTag("VOIP", null) + + val computedValue: String + get() { + if (parentTag == null) return value + return "${parentTag.computedValue}/$value" + } +} + diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt index 473adeb8d2..0fb7ce5307 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt @@ -20,11 +20,14 @@ import io.realm.Realm import org.matrix.android.sdk.api.session.events.model.Event import org.matrix.android.sdk.api.session.events.model.EventType import org.matrix.android.sdk.internal.database.model.EventInsertType +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.internal.session.EventInsertLiveProcessor import org.matrix.android.sdk.internal.session.SessionScope import timber.log.Timber import javax.inject.Inject +private val CallEventProcessorTag = LoggerTag("CallEventProcessor", LoggerTag.VOIP) + @SessionScope internal class CallEventProcessor @Inject constructor(private val callSignalingHandler: CallSignalingHandler) : EventInsertLiveProcessor { @@ -71,14 +74,8 @@ internal class CallEventProcessor @Inject constructor(private val callSignalingH } private fun dispatchToCallSignalingHandlerIfNeeded(event: Event) { - val now = System.currentTimeMillis() event.roomId ?: return Unit.also { - Timber.w("Event with no room id ${event.eventId}") - } - val age = now - (event.ageLocalTs ?: now) - if (age > 40_000) { - // Too old to ring? - return + Timber.tag(CallEventProcessorTag.computedValue).w("Event with no room id ${event.eventId}") } callSignalingHandler.onCallEvent(event) } diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt index b0901af719..18abfef5f8 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt @@ -32,10 +32,13 @@ import org.matrix.android.sdk.api.session.room.model.call.CallRejectContent import org.matrix.android.sdk.api.session.room.model.call.CallSelectAnswerContent import org.matrix.android.sdk.api.session.room.model.call.CallSignalingContent import org.matrix.android.sdk.internal.di.UserId +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.internal.session.SessionScope import timber.log.Timber import javax.inject.Inject +private val CallSignalingHandlerTag = LoggerTag("CallSignalingHandler", LoggerTag.VOIP) + @SessionScope internal class CallSignalingHandler @Inject constructor(private val activeCallHandler: ActiveCallHandler, private val mxCallFactory: MxCallFactory, @@ -111,12 +114,12 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa return } if (call.isOutgoing) { - Timber.v("Got selectAnswer for an outbound call: ignoring") + Timber.tag(CallSignalingHandlerTag.computedValue).v("Got selectAnswer for an outbound call: ignoring") return } val selectedPartyId = content.selectedPartyId if (selectedPartyId == null) { - Timber.w("Got nonsensical select_answer with null selected_party_id: ignoring") + Timber.tag(CallSignalingHandlerTag.computedValue).w("Got nonsensical select_answer with null selected_party_id: ignoring") return } callListenersDispatcher.onCallSelectAnswerReceived(content) @@ -130,7 +133,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa return } if (call.opponentPartyId != null && !call.partyIdsMatches(content)) { - Timber.v("Ignoring candidates from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") + Timber.tag(CallSignalingHandlerTag.computedValue).v("Ignoring candidates from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") return } callListenersDispatcher.onCallIceCandidateReceived(call, content) @@ -163,7 +166,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa // party ID must match (our chosen partner hanging up the call) or be undefined (we haven't chosen // a partner yet but we're treating the hangup as a reject as per VoIP v0) if (call.opponentPartyId != null && !call.partyIdsMatches(content)) { - Timber.v("Ignoring hangup from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") + Timber.tag(CallSignalingHandlerTag.computedValue).v("Ignoring hangup from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") return } if (call.state != CallState.Terminated) { @@ -180,12 +183,18 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa if (event.roomId == null || event.senderId == null) { return } + val now = System.currentTimeMillis() + val age = now - (event.ageLocalTs ?: now) + if (age > 40_000 && event.getClearType() == EventType.CALL_INVITE) { + Timber.tag(CallSignalingHandlerTag.computedValue).w("Call invite is too old to ring.") + return + } val content = event.getClearContent().toModel() ?: return content.callId ?: return if (invitedCallIds.contains(content.callId)) { // Call is already known, maybe due to fast lane. Ignore - Timber.d("Ignoring already known call invite") + Timber.tag(CallSignalingHandlerTag.computedValue).d("Ignoring already known call invite") return } val incomingCall = mxCallFactory.createIncomingCall( @@ -214,7 +223,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa callListenersDispatcher.onCallManagedByOtherSession(content.callId) } else { if (call.opponentPartyId != null) { - Timber.v("Ignoring answer from party ID ${content.partyId} we already have an answer from ${call.opponentPartyId}") + Timber.tag(CallSignalingHandlerTag.computedValue).v("Ignoring answer from party ID ${content.partyId} we already have an answer from ${call.opponentPartyId}") return } mxCallFactory.updateOutgoingCallWithOpponentData(call, event.senderId, content, content.capabilities) @@ -231,7 +240,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa activeCallHandler.getCallWithId(it) } if (currentCall == null) { - Timber.v("Call with id $callId is null") + Timber.tag(CallSignalingHandlerTag.computedValue).v("Call with id $callId is null") } return currentCall } diff --git a/vector/src/main/java/im/vector/app/core/services/CallService.kt b/vector/src/main/java/im/vector/app/core/services/CallService.kt index 59eee14d37..d83c3e47f5 100644 --- a/vector/src/main/java/im/vector/app/core/services/CallService.kt +++ b/vector/src/main/java/im/vector/app/core/services/CallService.kt @@ -37,9 +37,12 @@ import im.vector.app.features.home.AvatarRenderer import im.vector.app.features.notifications.NotificationUtils import im.vector.app.features.popup.IncomingCallAlert import im.vector.app.features.popup.PopupAlertManager +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.util.MatrixItem import timber.log.Timber +private val CallServiceTag = LoggerTag("CallService", LoggerTag.VOIP) + /** * Foreground service to manage calls */ @@ -91,7 +94,7 @@ class CallService : VectorService() { } override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { - Timber.v("## VOIP onStartCommand $intent") + Timber.tag(CallServiceTag.computedValue).v("onStartCommand $intent") if (mediaSession == null) { mediaSession = MediaSessionCompat(applicationContext, CallService::class.java.name).apply { setCallback(mediaSessionButtonCallback) @@ -115,19 +118,19 @@ class CallService : VectorService() { callRingPlayerOutgoing?.start() displayOutgoingRingingCallNotification(intent) } - ACTION_ONGOING_CALL -> { + ACTION_ONGOING_CALL -> { callRingPlayerIncoming?.stop() callRingPlayerOutgoing?.stop() displayCallInProgressNotification(intent) } - ACTION_CALL_CONNECTING -> { + ACTION_CALL_CONNECTING -> { // lower notification priority displayCallInProgressNotification(intent) // stop ringing callRingPlayerIncoming?.stop() callRingPlayerOutgoing?.stop() } - ACTION_CALL_TERMINATED -> { + ACTION_CALL_TERMINATED -> { handleCallTerminated(intent) } else -> { @@ -148,7 +151,7 @@ class CallService : VectorService() { * */ private fun displayIncomingCallNotification(intent: Intent) { - Timber.v("## VOIP displayIncomingCallNotification $intent") + Timber.tag(CallServiceTag.computedValue).v("displayIncomingCallNotification $intent") val callId = intent.getStringExtra(EXTRA_CALL_ID) ?: "" val call = callManager.getCallById(callId) ?: return Unit.also { handleUnexpectedState(callId) @@ -156,7 +159,7 @@ class CallService : VectorService() { val isVideoCall = call.mxCall.isVideoCall val fromBg = intent.getBooleanExtra(EXTRA_IS_IN_BG, false) val opponentMatrixItem = getOpponentMatrixItem(call) - Timber.v("displayIncomingCallNotification : display the dedicated notification") + Timber.tag(CallServiceTag.computedValue).v("displayIncomingCallNotification : display the dedicated notification") val incomingCallAlert = IncomingCallAlert(callId, shouldBeDisplayedIn = { activity -> if (activity is VectorCallActivity) { @@ -192,7 +195,7 @@ class CallService : VectorService() { val callId = intent.getStringExtra(EXTRA_CALL_ID) ?: "" alertManager.cancelAlert(callId) if (!knownCalls.remove(callId)) { - Timber.v("Call terminated for unknown call $callId$") + Timber.tag(CallServiceTag.computedValue).v("Call terminated for unknown call $callId$") handleUnexpectedState(callId) return } @@ -219,7 +222,7 @@ class CallService : VectorService() { handleUnexpectedState(callId) } val opponentMatrixItem = getOpponentMatrixItem(call) - Timber.v("displayOutgoingCallNotification : display the dedicated notification") + Timber.tag(CallServiceTag.computedValue).v("displayOutgoingCallNotification : display the dedicated notification") val notification = notificationUtils.buildOutgoingRingingCallNotification( call = call, title = opponentMatrixItem?.getBestName() ?: call.mxCall.opponentUserId @@ -236,7 +239,7 @@ class CallService : VectorService() { * Display a call in progress notification. */ private fun displayCallInProgressNotification(intent: Intent) { - Timber.v("## VOIP displayCallInProgressNotification") + Timber.tag(CallServiceTag.computedValue).v("displayCallInProgressNotification") val callId = intent.getStringExtra(EXTRA_CALL_ID) ?: "" val call = callManager.getCallById(callId) ?: return Unit.also { handleUnexpectedState(callId) @@ -256,7 +259,7 @@ class CallService : VectorService() { } private fun handleUnexpectedState(callId: String?) { - Timber.v("Fallback to clear everything") + Timber.tag(CallServiceTag.computedValue).v("Fallback to clear everything") callRingPlayerIncoming?.stop() callRingPlayerOutgoing?.stop() if (callId != null) { diff --git a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt index 3259b0915f..9bcd442be6 100644 --- a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt +++ b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt @@ -45,6 +45,7 @@ import kotlinx.coroutines.launch import kotlinx.coroutines.withContext import org.matrix.android.sdk.api.extensions.orFalse import org.matrix.android.sdk.api.extensions.tryOrNull +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.session.Session import org.matrix.android.sdk.api.session.call.CallIdGenerator import org.matrix.android.sdk.api.session.call.CallState @@ -88,6 +89,8 @@ private const val AUDIO_TRACK_ID = "${STREAM_ID}a0" private const val VIDEO_TRACK_ID = "${STREAM_ID}v0" private val DEFAULT_AUDIO_CONSTRAINTS = MediaConstraints() +private val WebRtcCallTag = LoggerTag("WebRtcCall", LoggerTag.VOIP) + class WebRtcCall( val mxCall: MxCall, // This is where the call is placed from an ui perspective. @@ -192,7 +195,7 @@ class WebRtcCall( .subscribe { // omit empty :/ if (it.isNotEmpty()) { - Timber.v("## Sending local ice candidates to call") + Timber.tag(WebRtcCallTag.computedValue).v("Sending local ice candidates to call") // it.forEach { peerConnection?.addIceCandidate(it) } mxCall.sendLocalCallCandidates(it.mapToCallCandidate()) } @@ -210,7 +213,7 @@ class WebRtcCall( fun onRenegotiationNeeded(restartIce: Boolean) { sessionScope?.launch(dispatcher) { if (mxCall.state != CallState.CreateOffer && mxCall.opponentVersion == 0) { - Timber.v("Opponent does not support renegotiation: ignoring onRenegotiationNeeded event") + Timber.tag(WebRtcCallTag.computedValue).v("Opponent does not support renegotiation: ignoring onRenegotiationNeeded event") return@launch } val constraints = MediaConstraints() @@ -218,7 +221,7 @@ class WebRtcCall( constraints.mandatory.add(MediaConstraints.KeyValuePair("IceRestart", "true")) } val peerConnection = peerConnection ?: return@launch - Timber.v("## VOIP creating offer...") + Timber.tag(WebRtcCallTag.computedValue).v("creating offer...") makingOffer = true try { val sessionDescription = peerConnection.awaitCreateOffer(constraints) ?: return@launch @@ -238,7 +241,7 @@ class WebRtcCall( } } catch (failure: Throwable) { // Need to handle error properly. - Timber.v("Failure while creating offer") + Timber.tag(WebRtcCallTag.computedValue).v("Failure while creating offer") } finally { makingOffer = false } @@ -267,7 +270,7 @@ class WebRtcCall( } } } - Timber.v("## VOIP creating peer connection...with iceServers $iceServers ") + Timber.tag(WebRtcCallTag.computedValue).v("creating peer connection...with iceServers $iceServers ") val rtcConfig = PeerConnection.RTCConfiguration(iceServers).apply { sdpSemantics = PeerConnection.SdpSemantics.UNIFIED_PLAN } @@ -314,7 +317,7 @@ class WebRtcCall( fun acceptIncomingCall() { sessionScope?.launch { - Timber.v("## VOIP acceptIncomingCall from state ${mxCall.state}") + Timber.tag(WebRtcCallTag.computedValue).v("acceptIncomingCall from state ${mxCall.state}") if (mxCall.state == CallState.LocalRinging) { internalAcceptIncomingCall() } @@ -333,7 +336,7 @@ class WebRtcCall( sender.dtmf()?.insertDtmf(digit, 100, 70) return@launch } catch (failure: Throwable) { - Timber.v("Fail to send Dtmf digit") + Timber.tag(WebRtcCallTag.computedValue).v("Fail to send Dtmf digit") } } } @@ -342,7 +345,7 @@ class WebRtcCall( fun attachViewRenderers(localViewRenderer: SurfaceViewRenderer?, remoteViewRenderer: SurfaceViewRenderer, mode: String?) { sessionScope?.launch(dispatcher) { - Timber.v("## VOIP attachViewRenderers localRendeder $localViewRenderer / $remoteViewRenderer") + Timber.tag(WebRtcCallTag.computedValue).v("attachViewRenderers localRendeder $localViewRenderer / $remoteViewRenderer") localSurfaceRenderers.addIfNeeded(localViewRenderer) remoteSurfaceRenderers.addIfNeeded(remoteViewRenderer) when (mode) { @@ -389,7 +392,7 @@ class WebRtcCall( } private suspend fun detachRenderersInternal(renderers: List?) = withContext(dispatcher) { - Timber.v("## VOIP detachRenderers") + Timber.tag(WebRtcCallTag.computedValue).v("detachRenderers") if (renderers.isNullOrEmpty()) { // remove all sinks localSurfaceRenderers.forEach { @@ -422,12 +425,12 @@ class WebRtcCall( // 2. Access camera (if video call) + microphone, create local stream createLocalStream() attachViewRenderersInternal() - Timber.v("## VOIP remoteCandidateSource $remoteCandidateSource") + Timber.tag(WebRtcCallTag.computedValue).v("remoteCandidateSource $remoteCandidateSource") remoteIceCandidateDisposable = remoteCandidateSource.subscribe({ - Timber.v("## VOIP adding remote ice candidate $it") + Timber.tag(WebRtcCallTag.computedValue).v("adding remote ice candidate $it") peerConnection?.addIceCandidate(it) }, { - Timber.v("## VOIP failed to add remote ice candidate $it") + Timber.tag(WebRtcCallTag.computedValue).v("failed to add remote ice candidate $it") }) // Now we wait for negotiation callback } @@ -453,14 +456,14 @@ class WebRtcCall( SessionDescription(SessionDescription.Type.OFFER, it) } if (offerSdp == null) { - Timber.v("We don't have any offer to process") + Timber.tag(WebRtcCallTag.computedValue).v("We don't have any offer to process") return@withContext } - Timber.v("Offer sdp for invite: ${offerSdp.description}") + Timber.tag(WebRtcCallTag.computedValue).v("Offer sdp for invite: ${offerSdp.description}") try { peerConnection?.awaitSetRemoteDescription(offerSdp) } catch (failure: Throwable) { - Timber.v("Failure putting remote description") + Timber.tag(WebRtcCallTag.computedValue).v("Failure putting remote description") endCall(true, CallHangupContent.Reason.UNKWOWN_ERROR) return@withContext } @@ -472,12 +475,12 @@ class WebRtcCall( createAnswer()?.also { mxCall.accept(it.description) } - Timber.v("## VOIP remoteCandidateSource $remoteCandidateSource") + Timber.tag(WebRtcCallTag.computedValue).v("remoteCandidateSource $remoteCandidateSource") remoteIceCandidateDisposable = remoteCandidateSource.subscribe({ - Timber.v("## VOIP adding remote ice candidate $it") + Timber.tag(WebRtcCallTag.computedValue).v("adding remote ice candidate $it") peerConnection?.addIceCandidate(it) }, { - Timber.v("## VOIP failed to add remote ice candidate $it") + Timber.tag(WebRtcCallTag.computedValue).v("failed to add remote ice candidate $it") }) } @@ -489,7 +492,7 @@ class WebRtcCall( private fun createLocalStream() { val peerConnectionFactory = peerConnectionFactoryProvider.get() ?: return - Timber.v("Create local stream for call ${mxCall.callId}") + Timber.tag(WebRtcCallTag.computedValue).v("Create local stream for call ${mxCall.callId}") configureAudioTrack(peerConnectionFactory) // add video track if needed if (mxCall.isVideoCall) { @@ -502,7 +505,7 @@ class WebRtcCall( val audioSource = peerConnectionFactory.createAudioSource(DEFAULT_AUDIO_CONSTRAINTS) val audioTrack = peerConnectionFactory.createAudioTrack(AUDIO_TRACK_ID, audioSource) audioTrack.setEnabled(true) - Timber.v("Add audio track $AUDIO_TRACK_ID to call ${mxCall.callId}") + Timber.tag(WebRtcCallTag.computedValue).v("Add audio track $AUDIO_TRACK_ID to call ${mxCall.callId}") peerConnection?.addTrack(audioTrack, listOf(STREAM_ID)) localAudioSource = audioSource localAudioTrack = audioTrack @@ -544,7 +547,7 @@ class WebRtcCall( override fun onCameraClosed() { super.onCameraClosed() - Timber.v("onCameraClosed") + Timber.tag(WebRtcCallTag.computedValue).v("onCameraClosed") // This could happen if you open the camera app in chat // We then register in order to restart capture as soon as the camera is available again videoCapturerIsInError = true @@ -552,16 +555,16 @@ class WebRtcCall( cameraAvailabilityCallback = object : CameraManager.AvailabilityCallback() { override fun onCameraUnavailable(cameraId: String) { super.onCameraUnavailable(cameraId) - Timber.v("On camera unavailable: $cameraId") + Timber.tag(WebRtcCallTag.computedValue).v("On camera unavailable: $cameraId") } override fun onCameraAccessPrioritiesChanged() { super.onCameraAccessPrioritiesChanged() - Timber.v("onCameraAccessPrioritiesChanged") + Timber.tag(WebRtcCallTag.computedValue).v("onCameraAccessPrioritiesChanged") } override fun onCameraAvailable(cameraId: String) { - Timber.v("On camera available: $cameraId") + Timber.tag(WebRtcCallTag.computedValue).v("On camera available: $cameraId") if (cameraId == camera.name) { videoCapturer?.startCapture(currentCaptureFormat.width, currentCaptureFormat.height, currentCaptureFormat.fps) cameraManager?.unregisterAvailabilityCallback(this) @@ -574,7 +577,7 @@ class WebRtcCall( val videoSource = peerConnectionFactory.createVideoSource(videoCapturer.isScreencast) val surfaceTextureHelper = SurfaceTextureHelper.create("CaptureThread", rootEglBase!!.eglBaseContext) - Timber.v("## VOIP Local video source created") + Timber.tag(WebRtcCallTag.computedValue).v("Local video source created") videoCapturer.initialize(surfaceTextureHelper, context, videoSource.capturerObserver) // HD @@ -582,7 +585,7 @@ class WebRtcCall( this.videoCapturer = videoCapturer val videoTrack = peerConnectionFactory.createVideoTrack(VIDEO_TRACK_ID, videoSource) - Timber.v("Add video track $VIDEO_TRACK_ID to call ${mxCall.callId}") + Timber.tag(WebRtcCallTag.computedValue).v("Add video track $VIDEO_TRACK_ID to call ${mxCall.callId}") videoTrack.setEnabled(true) peerConnection?.addTrack(videoTrack, listOf(STREAM_ID)) localVideoSource = videoSource @@ -592,7 +595,7 @@ class WebRtcCall( fun setCaptureFormat(format: CaptureFormat) { sessionScope?.launch(dispatcher) { - Timber.v("## VOIP setCaptureFormat $format") + Timber.tag(WebRtcCallTag.computedValue).v("setCaptureFormat $format") videoCapturer?.changeCaptureFormat(format.width, format.height, format.fps) currentCaptureFormat = format } @@ -686,14 +689,14 @@ class WebRtcCall( fun switchCamera() { sessionScope?.launch(dispatcher) { - Timber.v("## VOIP switchCamera") + Timber.tag(WebRtcCallTag.computedValue).v("switchCamera") if (mxCall.state is CallState.Connected && mxCall.isVideoCall) { val oppositeCamera = getOppositeCameraIfAny() ?: return@launch videoCapturer?.switchCamera( object : CameraVideoCapturer.CameraSwitchHandler { // Invoked on success. |isFrontCamera| is true if the new camera is front facing. override fun onCameraSwitchDone(isFrontCamera: Boolean) { - Timber.v("## VOIP onCameraSwitchDone isFront $isFrontCamera") + Timber.tag(WebRtcCallTag.computedValue).v("onCameraSwitchDone isFront $isFrontCamera") cameraInUse = oppositeCamera localSurfaceRenderers.forEach { it.get()?.setMirror(isFrontCamera) @@ -704,7 +707,7 @@ class WebRtcCall( } override fun onCameraSwitchError(errorDescription: String?) { - Timber.v("## VOIP onCameraSwitchError isFront $errorDescription") + Timber.tag(WebRtcCallTag.computedValue).v("onCameraSwitchError isFront $errorDescription") } }, oppositeCamera.name ) @@ -713,7 +716,7 @@ class WebRtcCall( } private suspend fun createAnswer(): SessionDescription? { - Timber.w("## VOIP createAnswer") + Timber.tag(WebRtcCallTag.computedValue).w("createAnswer") val peerConnection = peerConnection ?: return null val constraints = MediaConstraints().apply { mandatory.add(MediaConstraints.KeyValuePair("OfferToReceiveAudio", "true")) @@ -724,7 +727,7 @@ class WebRtcCall( peerConnection.awaitSetLocalDescription(localDescription) localDescription } catch (failure: Throwable) { - Timber.v("Fail to create answer") + Timber.tag(WebRtcCallTag.computedValue).v("Fail to create answer") null } } @@ -765,7 +768,7 @@ class WebRtcCall( sessionScope?.launch(dispatcher) { // reportError("Weird-looking stream: " + stream); if (stream.audioTracks.size > 1 || stream.videoTracks.size > 1) { - Timber.e("## VOIP StreamObserver weird looking stream: $stream") + Timber.tag(WebRtcCallTag.computedValue).e("## VOIP StreamObserver weird looking stream: $stream") // TODO maybe do something more?? endCall(true) return@launch @@ -829,7 +832,7 @@ class WebRtcCall( if (it.sdpMid.isNullOrEmpty() || it.candidate.isNullOrEmpty()) { return@forEach } - Timber.v("## VOIP onCallIceCandidateReceived for call ${mxCall.callId} sdp: ${it.candidate}") + Timber.tag(WebRtcCallTag.computedValue).v("onCallIceCandidateReceived for call ${mxCall.callId} sdp: ${it.candidate}") val iceCandidate = IceCandidate(it.sdpMid, it.sdpMLineIndex, it.candidate) remoteCandidateSource.onNext(iceCandidate) } @@ -838,7 +841,7 @@ class WebRtcCall( fun onCallAnswerReceived(callAnswerContent: CallAnswerContent) { sessionScope?.launch(dispatcher) { - Timber.v("## VOIP onCallAnswerReceived ${callAnswerContent.callId}") + Timber.tag(WebRtcCallTag.computedValue).v("onCallAnswerReceived ${callAnswerContent.callId}") val sdp = SessionDescription(SessionDescription.Type.ANSWER, callAnswerContent.answer.sdp) try { peerConnection?.awaitSetRemoteDescription(sdp) @@ -858,7 +861,7 @@ class WebRtcCall( val type = description?.type val sdpText = description?.sdp if (type == null || sdpText == null) { - Timber.i("Ignoring invalid m.call.negotiate event") + Timber.tag(WebRtcCallTag.computedValue).i("Ignoring invalid m.call.negotiate event") return@launch } val peerConnection = peerConnection ?: return@launch @@ -873,7 +876,7 @@ class WebRtcCall( ignoreOffer = !polite && offerCollision if (ignoreOffer) { - Timber.i("Ignoring colliding negotiate event because we're impolite") + Timber.tag(WebRtcCallTag.computedValue).i("Ignoring colliding negotiate event because we're impolite") return@launch } val prevOnHold = computeIsLocalOnHold() @@ -886,7 +889,7 @@ class WebRtcCall( } } } catch (failure: Throwable) { - Timber.e(failure, "Failed to complete negotiation") + Timber.tag(WebRtcCallTag.computedValue).e(failure, "Failed to complete negotiation") } val nowOnHold = computeIsLocalOnHold() wasLocalOnHold = nowOnHold @@ -909,7 +912,7 @@ class WebRtcCall( val session = sessionProvider.get() ?: return@launch val newAssertedIdentity = callAssertedIdentityContent.assertedIdentity ?: return@launch if (newAssertedIdentity.id == null && newAssertedIdentity.displayName == null) { - Timber.v("Asserted identity received with no relevant information, skip") + Timber.tag(WebRtcCallTag.computedValue).v("Asserted identity received with no relevant information, skip") return@launch } remoteAssertedIdentity = newAssertedIdentity diff --git a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt index 25463428e9..566ba7e81a 100644 --- a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt +++ b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt @@ -33,6 +33,7 @@ import im.vector.app.push.fcm.FcmHelper import kotlinx.coroutines.asCoroutineDispatcher import org.matrix.android.sdk.api.extensions.orFalse import org.matrix.android.sdk.api.extensions.tryOrNull +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.session.Session import org.matrix.android.sdk.api.session.call.CallListener import org.matrix.android.sdk.api.session.call.CallState @@ -60,6 +61,8 @@ import javax.inject.Singleton * Manage peerConnectionFactory & Peer connections outside of activity lifecycle to resist configuration changes * Use app context */ +private val WebRtcCallManagerTag = LoggerTag("WebRtcCallManager", LoggerTag.VOIP) + @Singleton class WebRtcCallManager @Inject constructor( private val context: Context, @@ -184,7 +187,7 @@ class WebRtcCallManager @Inject constructor( fun getAdvertisedCalls() = advertisedCalls fun headSetButtonTapped() { - Timber.v("## VOIP headSetButtonTapped") + Timber.tag(WebRtcCallManagerTag.computedValue).v("headSetButtonTapped") val call = getCurrentCall() ?: return if (call.mxCall.state is CallState.LocalRinging) { call.acceptIncomingCall() @@ -197,12 +200,12 @@ class WebRtcCallManager @Inject constructor( private fun createPeerConnectionFactoryIfNeeded() { if (peerConnectionFactory != null) return - Timber.v("## VOIP createPeerConnectionFactory") + Timber.tag(WebRtcCallManagerTag.computedValue).v("createPeerConnectionFactory") val eglBaseContext = rootEglBase?.eglBaseContext ?: return Unit.also { - Timber.e("## VOIP No EGL BASE") + Timber.tag(WebRtcCallManagerTag.computedValue).e("No EGL BASE") } - Timber.v("## VOIP PeerConnectionFactory.initialize") + Timber.tag(WebRtcCallManagerTag.computedValue).v("PeerConnectionFactory.initialize") PeerConnectionFactory.initialize(PeerConnectionFactory .InitializationOptions.builder(context.applicationContext) .createInitializationOptions() @@ -216,7 +219,7 @@ class WebRtcCallManager @Inject constructor( /* enableH264HighProfile */ true) val defaultVideoDecoderFactory = DefaultVideoDecoderFactory(eglBaseContext) - Timber.v("## VOIP PeerConnectionFactory.createPeerConnectionFactory ...") + Timber.tag(WebRtcCallManagerTag.computedValue).v("PeerConnectionFactory.createPeerConnectionFactory ...") peerConnectionFactory = PeerConnectionFactory.builder() .setOptions(options) .setVideoEncoderFactory(defaultVideoEncoderFactory) @@ -225,7 +228,7 @@ class WebRtcCallManager @Inject constructor( } private fun onCallActive(call: WebRtcCall) { - Timber.v("## VOIP WebRtcPeerConnectionManager onCall active: ${call.mxCall.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).v("WebRtcPeerConnectionManager onCall active: ${call.mxCall.callId}") val currentCall = getCurrentCall().takeIf { it != call } currentCall?.updateRemoteOnHold(onHold = true) audioManager.setMode(if (call.mxCall.isVideoCall) CallAudioManager.Mode.VIDEO_CALL else CallAudioManager.Mode.AUDIO_CALL) @@ -233,9 +236,9 @@ class WebRtcCallManager @Inject constructor( } private fun onCallEnded(callId: String) { - Timber.v("## VOIP WebRtcPeerConnectionManager onCall ended: $callId") + Timber.tag(WebRtcCallManagerTag.computedValue).v("WebRtcPeerConnectionManager onCall ended: $callId") val webRtcCall = callsByCallId.remove(callId) ?: return Unit.also { - Timber.v("On call ended for unknown call $callId") + Timber.tag(WebRtcCallManagerTag.computedValue).v("On call ended for unknown call $callId") } CallService.onCallTerminated(context, callId) callsByRoomId[webRtcCall.signalingRoomId]?.remove(webRtcCall) @@ -247,7 +250,7 @@ class WebRtcCallManager @Inject constructor( } // There is no active calls if (getCurrentCall() == null) { - Timber.v("## VOIP Dispose peerConnectionFactory as there is no need to keep one") + Timber.tag(WebRtcCallManagerTag.computedValue).v("Dispose peerConnectionFactory as there is no need to keep one") peerConnectionFactory?.dispose() peerConnectionFactory = null audioManager.setMode(CallAudioManager.Mode.DEFAULT) @@ -265,13 +268,13 @@ class WebRtcCallManager @Inject constructor( suspend fun startOutgoingCall(nativeRoomId: String, otherUserId: String, isVideoCall: Boolean, transferee: WebRtcCall? = null) { val signalingRoomId = callUserMapper?.getOrCreateVirtualRoomForRoom(nativeRoomId, otherUserId) ?: nativeRoomId - Timber.v("## VOIP startOutgoingCall in room $signalingRoomId to $otherUserId isVideo $isVideoCall") + Timber.tag(WebRtcCallManagerTag.computedValue).v("startOutgoingCall in room $signalingRoomId to $otherUserId isVideo $isVideoCall") if (getCallsByRoomId(nativeRoomId).isNotEmpty()) { - Timber.w("## VOIP you already have a call in this room") + Timber.tag(WebRtcCallManagerTag.computedValue).w("you already have a call in this room") return } if (getCurrentCall() != null && getCurrentCall()?.mxCall?.state !is CallState.Connected || getCalls().size >= 2) { - Timber.w("## VOIP cannot start outgoing call") + Timber.tag(WebRtcCallManagerTag.computedValue).w("cannot start outgoing call") // Just ignore, maybe we could answer from other session? return } @@ -294,10 +297,10 @@ class WebRtcCallManager @Inject constructor( } override fun onCallIceCandidateReceived(mxCall: MxCall, iceCandidatesContent: CallCandidatesContent) { - Timber.v("## VOIP onCallIceCandidateReceived for call ${mxCall.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).v("onCallIceCandidateReceived for call ${mxCall.callId}") val call = callsByCallId[iceCandidatesContent.callId] ?: return Unit.also { - Timber.w("onCallIceCandidateReceived for non active call? ${iceCandidatesContent.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallIceCandidateReceived for non active call? ${iceCandidatesContent.callId}") } call.onCallIceCandidateReceived(iceCandidatesContent) } @@ -334,14 +337,14 @@ class WebRtcCallManager @Inject constructor( } override fun onCallInviteReceived(mxCall: MxCall, callInviteContent: CallInviteContent) { - Timber.v("## VOIP onCallInviteReceived callId ${mxCall.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).v("onCallInviteReceived callId ${mxCall.callId}") val nativeRoomId = callUserMapper?.nativeRoomForVirtualRoom(mxCall.roomId) ?: mxCall.roomId if (getCallsByRoomId(nativeRoomId).isNotEmpty()) { - Timber.w("## VOIP you already have a call in this room") + Timber.tag(WebRtcCallManagerTag.computedValue).w("you already have a call in this room") return } if ((getCurrentCall() != null && getCurrentCall()?.mxCall?.state !is CallState.Connected) || getCalls().size >= 2) { - Timber.w("## VOIP receiving incoming call but cannot handle it") + Timber.tag(WebRtcCallManagerTag.computedValue).w("receiving incoming call but cannot handle it") // Just ignore, maybe we could answer from other session? return } @@ -370,7 +373,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallAnswerReceived(callAnswerContent: CallAnswerContent) { val call = callsByCallId[callAnswerContent.callId] ?: return Unit.also { - Timber.w("onCallAnswerReceived for non active call? ${callAnswerContent.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallAnswerReceived for non active call? ${callAnswerContent.callId}") } val mxCall = call.mxCall // Update service state @@ -384,7 +387,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallHangupReceived(callHangupContent: CallHangupContent) { val call = callsByCallId[callHangupContent.callId] ?: return Unit.also { - Timber.w("onCallHangupReceived for non active call? ${callHangupContent.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallHangupReceived for non active call? ${callHangupContent.callId}") } call.endCall(false) } @@ -392,7 +395,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallRejectReceived(callRejectContent: CallRejectContent) { val call = callsByCallId[callRejectContent.callId] ?: return Unit.also { - Timber.w("onCallRejectReceived for non active call? ${callRejectContent.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallRejectReceived for non active call? ${callRejectContent.callId}") } call.endCall(false) } @@ -400,7 +403,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallSelectAnswerReceived(callSelectAnswerContent: CallSelectAnswerContent) { val call = callsByCallId[callSelectAnswerContent.callId] ?: return Unit.also { - Timber.w("onCallSelectAnswerReceived for non active call? ${callSelectAnswerContent.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallSelectAnswerReceived for non active call? ${callSelectAnswerContent.callId}") } val selectedPartyId = callSelectAnswerContent.selectedPartyId if (selectedPartyId != call.mxCall.ourPartyId) { @@ -413,13 +416,13 @@ class WebRtcCallManager @Inject constructor( override fun onCallNegotiateReceived(callNegotiateContent: CallNegotiateContent) { val call = callsByCallId[callNegotiateContent.callId] ?: return Unit.also { - Timber.w("onCallNegotiateReceived for non active call? ${callNegotiateContent.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallNegotiateReceived for non active call? ${callNegotiateContent.callId}") } call.onCallNegotiateReceived(callNegotiateContent) } override fun onCallManagedByOtherSession(callId: String) { - Timber.v("## VOIP onCallManagedByOtherSession: $callId") + Timber.tag(WebRtcCallManagerTag.computedValue).v("onCallManagedByOtherSession: $callId") onCallEnded(callId) } @@ -429,8 +432,8 @@ class WebRtcCallManager @Inject constructor( } val call = callsByCallId[callAssertedIdentityContent.callId] ?: return Unit.also { - Timber.w("onCallAssertedIdentityReceived for non active call? ${callAssertedIdentityContent.callId}") + Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallAssertedIdentityReceived for non active call? ${callAssertedIdentityContent.callId}") } - call.onCallAssertedIdentityReceived(callAssertedIdentityContent) + call.onCallAssertedIdentityReceived(callAssertedIdentityContent) } } From 79c8ef7ebf0c7e16cc34036631c9172b4db16885 Mon Sep 17 00:00:00 2001 From: ganfra Date: Thu, 22 Jul 2021 15:22:15 +0200 Subject: [PATCH 2/4] LoggerTag: try new thing and branch more on VOIP --- .../android/sdk/api/logger/LoggerTag.kt | 17 ++-- .../session/call/CallEventProcessor.kt | 4 +- .../session/call/CallSignalingHandler.kt | 21 ++--- .../call/DefaultCallSignalingService.kt | 2 - .../internal/session/call/model/MxCallImpl.kt | 23 +++--- .../vector/app/core/services/CallService.kt | 16 ++-- .../app/features/call/VectorCallActivity.kt | 19 +++-- .../call/audio/API21AudioDeviceDetector.kt | 21 ++--- .../call/webrtc/PeerConnectionObserver.kt | 27 ++++--- .../webrtc/PeerConnectionObserverAdapter.kt | 70 ---------------- .../app/features/call/webrtc/WebRtcCall.kt | 80 +++++++++---------- .../features/call/webrtc/WebRtcCallManager.kt | 50 ++++++------ 12 files changed, 148 insertions(+), 202 deletions(-) delete mode 100644 vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserverAdapter.kt diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt index 074bd5cf6c..9978ae32aa 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt @@ -17,16 +17,21 @@ package org.matrix.android.sdk.api.logger /** - * Some tags used on Timber. + * Parent class for custom logger tags. Can be used with Timber : + * + * val loggerTag = LoggerTag("MyTag", LoggerTag.VOIP) + * Timber.tag(loggerTag.value).v("My log message") */ -open class LoggerTag(private val value: String, private val parentTag: LoggerTag? = null) { +open class LoggerTag(private val _value: String, private val parentTag: LoggerTag? = null) { object VOIP : LoggerTag("VOIP", null) - val computedValue: String + val value: String get() { - if (parentTag == null) return value - return "${parentTag.computedValue}/$value" + return if (parentTag == null) { + _value + } else { + "${parentTag.value}/$_value" + } } } - diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt index 0fb7ce5307..bdc254fc99 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallEventProcessor.kt @@ -26,7 +26,7 @@ import org.matrix.android.sdk.internal.session.SessionScope import timber.log.Timber import javax.inject.Inject -private val CallEventProcessorTag = LoggerTag("CallEventProcessor", LoggerTag.VOIP) +private val loggerTag = LoggerTag("CallEventProcessor", LoggerTag.VOIP) @SessionScope internal class CallEventProcessor @Inject constructor(private val callSignalingHandler: CallSignalingHandler) @@ -75,7 +75,7 @@ internal class CallEventProcessor @Inject constructor(private val callSignalingH private fun dispatchToCallSignalingHandlerIfNeeded(event: Event) { event.roomId ?: return Unit.also { - Timber.tag(CallEventProcessorTag.computedValue).w("Event with no room id ${event.eventId}") + Timber.tag(loggerTag.value).w("Event with no room id ${event.eventId}") } callSignalingHandler.onCallEvent(event) } diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt index 18abfef5f8..17e9aa9b7f 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt @@ -16,6 +16,7 @@ package org.matrix.android.sdk.internal.session.call +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.session.call.CallListener import org.matrix.android.sdk.api.session.call.CallState import org.matrix.android.sdk.api.session.call.MxCall @@ -32,12 +33,11 @@ import org.matrix.android.sdk.api.session.room.model.call.CallRejectContent import org.matrix.android.sdk.api.session.room.model.call.CallSelectAnswerContent import org.matrix.android.sdk.api.session.room.model.call.CallSignalingContent import org.matrix.android.sdk.internal.di.UserId -import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.internal.session.SessionScope import timber.log.Timber import javax.inject.Inject -private val CallSignalingHandlerTag = LoggerTag("CallSignalingHandler", LoggerTag.VOIP) +private val loggerTag = LoggerTag("CallSignalingHandler", LoggerTag.VOIP) @SessionScope internal class CallSignalingHandler @Inject constructor(private val activeCallHandler: ActiveCallHandler, @@ -114,12 +114,12 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa return } if (call.isOutgoing) { - Timber.tag(CallSignalingHandlerTag.computedValue).v("Got selectAnswer for an outbound call: ignoring") + Timber.tag(loggerTag.value).v("Got selectAnswer for an outbound call: ignoring") return } val selectedPartyId = content.selectedPartyId if (selectedPartyId == null) { - Timber.tag(CallSignalingHandlerTag.computedValue).w("Got nonsensical select_answer with null selected_party_id: ignoring") + Timber.tag(loggerTag.value).w("Got nonsensical select_answer with null selected_party_id: ignoring") return } callListenersDispatcher.onCallSelectAnswerReceived(content) @@ -133,7 +133,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa return } if (call.opponentPartyId != null && !call.partyIdsMatches(content)) { - Timber.tag(CallSignalingHandlerTag.computedValue).v("Ignoring candidates from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") + Timber.tag(loggerTag.value).v("Ignoring candidates from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") return } callListenersDispatcher.onCallIceCandidateReceived(call, content) @@ -166,7 +166,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa // party ID must match (our chosen partner hanging up the call) or be undefined (we haven't chosen // a partner yet but we're treating the hangup as a reject as per VoIP v0) if (call.opponentPartyId != null && !call.partyIdsMatches(content)) { - Timber.tag(CallSignalingHandlerTag.computedValue).v("Ignoring hangup from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") + Timber.tag(loggerTag.value).v("Ignoring hangup from party ID ${content.partyId} we have chosen party ID ${call.opponentPartyId}") return } if (call.state != CallState.Terminated) { @@ -186,7 +186,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa val now = System.currentTimeMillis() val age = now - (event.ageLocalTs ?: now) if (age > 40_000 && event.getClearType() == EventType.CALL_INVITE) { - Timber.tag(CallSignalingHandlerTag.computedValue).w("Call invite is too old to ring.") + Timber.tag(loggerTag.value).w("Call invite is too old to ring.") return } val content = event.getClearContent().toModel() ?: return @@ -194,7 +194,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa content.callId ?: return if (invitedCallIds.contains(content.callId)) { // Call is already known, maybe due to fast lane. Ignore - Timber.tag(CallSignalingHandlerTag.computedValue).d("Ignoring already known call invite") + Timber.tag(loggerTag.value).d("Ignoring already known call invite") return } val incomingCall = mxCallFactory.createIncomingCall( @@ -223,7 +223,8 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa callListenersDispatcher.onCallManagedByOtherSession(content.callId) } else { if (call.opponentPartyId != null) { - Timber.tag(CallSignalingHandlerTag.computedValue).v("Ignoring answer from party ID ${content.partyId} we already have an answer from ${call.opponentPartyId}") + Timber.tag(loggerTag.value) + .v("Ignoring answer from party ID ${content.partyId} we already have an answer from ${call.opponentPartyId}") return } mxCallFactory.updateOutgoingCallWithOpponentData(call, event.senderId, content, content.capabilities) @@ -240,7 +241,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa activeCallHandler.getCallWithId(it) } if (currentCall == null) { - Timber.tag(CallSignalingHandlerTag.computedValue).v("Call with id $callId is null") + Timber.tag(loggerTag.value).v("Call with id $callId is null") } return currentCall } diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/DefaultCallSignalingService.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/DefaultCallSignalingService.kt index da1f84cc89..4a949e21a6 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/DefaultCallSignalingService.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/DefaultCallSignalingService.kt @@ -21,7 +21,6 @@ import org.matrix.android.sdk.api.session.call.CallSignalingService import org.matrix.android.sdk.api.session.call.MxCall import org.matrix.android.sdk.api.session.call.TurnServerResponse import org.matrix.android.sdk.internal.session.SessionScope -import timber.log.Timber import javax.inject.Inject @SessionScope @@ -51,7 +50,6 @@ internal class DefaultCallSignalingService @Inject constructor( } override fun getCallWithId(callId: String): MxCall? { - Timber.v("## VOIP getCallWithId $callId all calls ${activeCallHandler.getActiveCallsLiveData().value?.map { it.callId }}") return activeCallHandler.getCallWithId(callId) } diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/model/MxCallImpl.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/model/MxCallImpl.kt index f101685a4b..1d831afcae 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/model/MxCallImpl.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/model/MxCallImpl.kt @@ -17,6 +17,7 @@ package org.matrix.android.sdk.internal.session.call.model import org.matrix.android.sdk.api.MatrixConfiguration +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.session.call.CallIdGenerator import org.matrix.android.sdk.api.session.call.CallState import org.matrix.android.sdk.api.session.call.MxCall @@ -47,6 +48,8 @@ import org.matrix.android.sdk.internal.session.room.send.queue.EventSenderProces import timber.log.Timber import java.math.BigDecimal +private val loggerTag = LoggerTag("MxCallImpl", LoggerTag.VOIP) + internal class MxCallImpl( override val callId: String, override val isOutgoing: Boolean, @@ -93,7 +96,7 @@ internal class MxCallImpl( try { it.onStateUpdate(this) } catch (failure: Throwable) { - Timber.d("dispatchStateChange failed for call $callId : ${failure.localizedMessage}") + Timber.tag(loggerTag.value).d("dispatchStateChange failed for call $callId : ${failure.localizedMessage}") } } } @@ -109,7 +112,7 @@ internal class MxCallImpl( override fun offerSdp(sdpString: String) { if (!isOutgoing) return - Timber.v("## VOIP offerSdp $callId") + Timber.tag(loggerTag.value).v("offerSdp $callId") state = CallState.Dialing CallInviteContent( callId = callId, @@ -124,7 +127,7 @@ internal class MxCallImpl( } override fun sendLocalCallCandidates(candidates: List) { - Timber.v("Send local call canditates $callId: $candidates") + Timber.tag(loggerTag.value).v("Send local call canditates $callId: $candidates") CallCandidatesContent( callId = callId, partyId = ourPartyId, @@ -141,11 +144,11 @@ internal class MxCallImpl( override fun reject() { if (opponentVersion < 1) { - Timber.v("Opponent version is less than 1 ($opponentVersion): sending hangup instead of reject") + Timber.tag(loggerTag.value).v("Opponent version is less than 1 ($opponentVersion): sending hangup instead of reject") hangUp() return } - Timber.v("## VOIP reject $callId") + Timber.tag(loggerTag.value).v("reject $callId") CallRejectContent( callId = callId, partyId = ourPartyId, @@ -157,7 +160,7 @@ internal class MxCallImpl( } override fun hangUp(reason: CallHangupContent.Reason?) { - Timber.v("## VOIP hangup $callId") + Timber.tag(loggerTag.value).v("hangup $callId") CallHangupContent( callId = callId, partyId = ourPartyId, @@ -170,7 +173,7 @@ internal class MxCallImpl( } override fun accept(sdpString: String) { - Timber.v("## VOIP accept $callId") + Timber.tag(loggerTag.value).v("accept $callId") if (isOutgoing) return state = CallState.Answering CallAnswerContent( @@ -185,7 +188,7 @@ internal class MxCallImpl( } override fun negotiate(sdpString: String, type: SdpType) { - Timber.v("## VOIP negotiate $callId") + Timber.tag(loggerTag.value).v("negotiate $callId") CallNegotiateContent( callId = callId, partyId = ourPartyId, @@ -198,7 +201,7 @@ internal class MxCallImpl( } override fun selectAnswer() { - Timber.v("## VOIP select answer $callId") + Timber.tag(loggerTag.value).v("select answer $callId") if (isOutgoing) return state = CallState.Answering CallSelectAnswerContent( @@ -219,7 +222,7 @@ internal class MxCallImpl( val profileInfo = try { getProfileInfoTask.execute(profileInfoParams) } catch (failure: Throwable) { - Timber.v("Fail fetching profile info of $targetUserId while transferring call") + Timber.tag(loggerTag.value).v("Fail fetching profile info of $targetUserId while transferring call") null } CallReplacesContent( diff --git a/vector/src/main/java/im/vector/app/core/services/CallService.kt b/vector/src/main/java/im/vector/app/core/services/CallService.kt index d83c3e47f5..23a57e459e 100644 --- a/vector/src/main/java/im/vector/app/core/services/CallService.kt +++ b/vector/src/main/java/im/vector/app/core/services/CallService.kt @@ -41,7 +41,7 @@ import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.util.MatrixItem import timber.log.Timber -private val CallServiceTag = LoggerTag("CallService", LoggerTag.VOIP) +private val loggerTag = LoggerTag("CallService", LoggerTag.VOIP) /** * Foreground service to manage calls @@ -94,7 +94,7 @@ class CallService : VectorService() { } override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int { - Timber.tag(CallServiceTag.computedValue).v("onStartCommand $intent") + Timber.tag(loggerTag.value).v("onStartCommand $intent") if (mediaSession == null) { mediaSession = MediaSessionCompat(applicationContext, CallService::class.java.name).apply { setCallback(mediaSessionButtonCallback) @@ -151,7 +151,7 @@ class CallService : VectorService() { * */ private fun displayIncomingCallNotification(intent: Intent) { - Timber.tag(CallServiceTag.computedValue).v("displayIncomingCallNotification $intent") + Timber.tag(loggerTag.value).v("displayIncomingCallNotification $intent") val callId = intent.getStringExtra(EXTRA_CALL_ID) ?: "" val call = callManager.getCallById(callId) ?: return Unit.also { handleUnexpectedState(callId) @@ -159,7 +159,7 @@ class CallService : VectorService() { val isVideoCall = call.mxCall.isVideoCall val fromBg = intent.getBooleanExtra(EXTRA_IS_IN_BG, false) val opponentMatrixItem = getOpponentMatrixItem(call) - Timber.tag(CallServiceTag.computedValue).v("displayIncomingCallNotification : display the dedicated notification") + Timber.tag(loggerTag.value).v("displayIncomingCallNotification : display the dedicated notification") val incomingCallAlert = IncomingCallAlert(callId, shouldBeDisplayedIn = { activity -> if (activity is VectorCallActivity) { @@ -195,7 +195,7 @@ class CallService : VectorService() { val callId = intent.getStringExtra(EXTRA_CALL_ID) ?: "" alertManager.cancelAlert(callId) if (!knownCalls.remove(callId)) { - Timber.tag(CallServiceTag.computedValue).v("Call terminated for unknown call $callId$") + Timber.tag(loggerTag.value).v("Call terminated for unknown call $callId$") handleUnexpectedState(callId) return } @@ -222,7 +222,7 @@ class CallService : VectorService() { handleUnexpectedState(callId) } val opponentMatrixItem = getOpponentMatrixItem(call) - Timber.tag(CallServiceTag.computedValue).v("displayOutgoingCallNotification : display the dedicated notification") + Timber.tag(loggerTag.value).v("displayOutgoingCallNotification : display the dedicated notification") val notification = notificationUtils.buildOutgoingRingingCallNotification( call = call, title = opponentMatrixItem?.getBestName() ?: call.mxCall.opponentUserId @@ -239,7 +239,7 @@ class CallService : VectorService() { * Display a call in progress notification. */ private fun displayCallInProgressNotification(intent: Intent) { - Timber.tag(CallServiceTag.computedValue).v("displayCallInProgressNotification") + Timber.tag(loggerTag.value).v("displayCallInProgressNotification") val callId = intent.getStringExtra(EXTRA_CALL_ID) ?: "" val call = callManager.getCallById(callId) ?: return Unit.also { handleUnexpectedState(callId) @@ -259,7 +259,7 @@ class CallService : VectorService() { } private fun handleUnexpectedState(callId: String?) { - Timber.tag(CallServiceTag.computedValue).v("Fallback to clear everything") + Timber.tag(loggerTag.value).v("Fallback to clear everything") callRingPlayerIncoming?.stop() callRingPlayerOutgoing?.stop() if (callId != null) { diff --git a/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt b/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt index 7e84811102..7a22ac3b8e 100644 --- a/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt +++ b/vector/src/main/java/im/vector/app/features/call/VectorCallActivity.kt @@ -54,6 +54,7 @@ import im.vector.app.features.home.room.detail.RoomDetailArgs import io.reactivex.android.schedulers.AndroidSchedulers import kotlinx.parcelize.Parcelize import org.matrix.android.sdk.api.extensions.orFalse +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.session.call.CallState import org.matrix.android.sdk.api.session.call.MxPeerConnectionState import org.matrix.android.sdk.api.session.call.TurnServerResponse @@ -71,6 +72,8 @@ data class CallArgs( val isVideoCall: Boolean ) : Parcelable +private val loggerTag = LoggerTag("VectorCallActivity", LoggerTag.VOIP) + class VectorCallActivity : VectorBaseActivity(), CallControlsView.InteractionListener { override fun getBinding() = ActivityCallBinding.inflate(layoutInflater) @@ -113,11 +116,11 @@ class VectorCallActivity : VectorBaseActivity(), CallContro if (intent.hasExtra(MvRx.KEY_ARG)) { callArgs = intent.getParcelableExtra(MvRx.KEY_ARG)!! } else { - Timber.e("## VOIP missing callArgs for VectorCall Activity") + Timber.tag(loggerTag.value).e("missing callArgs for VectorCall Activity") finish() } - Timber.v("## VOIP EXTRA_MODE is ${intent.getStringExtra(EXTRA_MODE)}") + Timber.tag(loggerTag.value).v("EXTRA_MODE is ${intent.getStringExtra(EXTRA_MODE)}") if (intent.getStringExtra(EXTRA_MODE) == INCOMING_RINGING) { turnScreenOnAndKeyguardOff() } @@ -160,7 +163,7 @@ class VectorCallActivity : VectorBaseActivity(), CallContro } private fun renderState(state: VectorCallViewState) { - Timber.v("## VOIP renderState call $state") + Timber.tag(loggerTag.value).v("renderState call $state") if (state.callState is Fail) { finish() return @@ -309,7 +312,7 @@ class VectorCallActivity : VectorBaseActivity(), CallContro private fun start() { rootEglBase = EglUtils.rootEglBase ?: return Unit.also { - Timber.v("## VOIP rootEglBase is null") + Timber.tag(loggerTag.value).v("rootEglBase is null") finish() } @@ -335,7 +338,7 @@ class VectorCallActivity : VectorBaseActivity(), CallContro } private fun handleViewEvents(event: VectorCallViewEvents?) { - Timber.v("## VOIP handleViewEvents $event") + Timber.tag(loggerTag.value).v("handleViewEvents $event") when (event) { VectorCallViewEvents.DismissNoCall -> { finish() @@ -357,7 +360,7 @@ class VectorCallActivity : VectorBaseActivity(), CallContro } private fun onErrorTimoutConnect(turn: TurnServerResponse?) { - Timber.d("## VOIP onErrorTimoutConnect $turn") + Timber.tag(loggerTag.value).d("onErrorTimoutConnect $turn") // TODO ask to use default stun, etc... MaterialAlertDialogBuilder(this) .setTitle(R.string.call_failed_no_connection) @@ -437,7 +440,7 @@ class VectorCallActivity : VectorBaseActivity(), CallContro // Needed to let you answer call when phone is locked private fun turnScreenOnAndKeyguardOff() { - Timber.v("## VOIP turnScreenOnAndKeyguardOff") + Timber.tag(loggerTag.value).v("turnScreenOnAndKeyguardOff") if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O_MR1) { setShowWhenLocked(true) setTurnScreenOn(true) @@ -458,7 +461,7 @@ class VectorCallActivity : VectorBaseActivity(), CallContro } private fun turnScreenOffAndKeyguardOn() { - Timber.v("## VOIP turnScreenOnAndKeyguardOn") + Timber.tag(loggerTag.value).v("turnScreenOnAndKeyguardOn") if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O_MR1) { setShowWhenLocked(false) setTurnScreenOn(false) diff --git a/vector/src/main/java/im/vector/app/features/call/audio/API21AudioDeviceDetector.kt b/vector/src/main/java/im/vector/app/features/call/audio/API21AudioDeviceDetector.kt index 32b243aa2b..4f54f703b4 100644 --- a/vector/src/main/java/im/vector/app/features/call/audio/API21AudioDeviceDetector.kt +++ b/vector/src/main/java/im/vector/app/features/call/audio/API21AudioDeviceDetector.kt @@ -25,9 +25,12 @@ import android.media.AudioManager import androidx.core.content.getSystemService import im.vector.app.core.services.BluetoothHeadsetReceiver import im.vector.app.core.services.WiredHeadsetStateReceiver +import org.matrix.android.sdk.api.logger.LoggerTag import timber.log.Timber import java.util.HashSet +private val loggerTag = LoggerTag("API21AudioDeviceDetector", LoggerTag.VOIP) + internal class API21AudioDeviceDetector(private val context: Context, private val audioManager: AudioManager, private val callAudioManager: CallAudioManager @@ -62,17 +65,17 @@ internal class API21AudioDeviceDetector(private val context: Context, } private fun isBluetoothHeadsetOn(): Boolean { - Timber.v("## VOIP: AudioManager isBluetoothHeadsetOn") + Timber.tag(loggerTag.value).v("AudioManager isBluetoothHeadsetOn") try { if (connectedBlueToothHeadset == null) return false.also { - Timber.v("## VOIP: AudioManager no connected bluetooth headset") + Timber.tag(loggerTag.value).v("AudioManager no connected bluetooth headset") } if (!audioManager.isBluetoothScoAvailableOffCall) return false.also { - Timber.v("## VOIP: AudioManager isBluetoothScoAvailableOffCall false") + Timber.tag(loggerTag.value).v("AudioManager isBluetoothScoAvailableOffCall false") } return true } catch (failure: Throwable) { - Timber.e("## VOIP: AudioManager isBluetoothHeadsetOn failure ${failure.localizedMessage}") + Timber.e("AudioManager isBluetoothHeadsetOn failure ${failure.localizedMessage}") return false } } @@ -91,11 +94,11 @@ internal class API21AudioDeviceDetector(private val context: Context, bluetoothHeadsetStateReceiver = BluetoothHeadsetReceiver.createAndRegister(context, this) val bm: BluetoothManager? = context.getSystemService() val adapter = bm?.adapter - Timber.d("## VOIP Bluetooth adapter $adapter") + Timber.tag(loggerTag.value).d("Bluetooth adapter $adapter") bluetoothAdapter = adapter adapter?.getProfileProxy(context, object : BluetoothProfile.ServiceListener { override fun onServiceDisconnected(profile: Int) { - Timber.d("## VOIP onServiceDisconnected $profile") + Timber.tag(loggerTag.value).d("onServiceDisconnected $profile") if (profile == BluetoothProfile.HEADSET) { connectedBlueToothHeadset = null onAudioDeviceChange() @@ -103,7 +106,7 @@ internal class API21AudioDeviceDetector(private val context: Context, } override fun onServiceConnected(profile: Int, proxy: BluetoothProfile?) { - Timber.d("## VOIP onServiceConnected $profile , proxy:$proxy") + Timber.tag(loggerTag.value).d("onServiceConnected $profile , proxy:$proxy") if (profile == BluetoothProfile.HEADSET) { connectedBlueToothHeadset = proxy onAudioDeviceChange() @@ -122,12 +125,12 @@ internal class API21AudioDeviceDetector(private val context: Context, } override fun onHeadsetEvent(event: WiredHeadsetStateReceiver.HeadsetPlugEvent) { - Timber.v("onHeadsetEvent $event") + Timber.tag(loggerTag.value).v("onHeadsetEvent $event") onAudioDeviceChange() } override fun onBTHeadsetEvent(event: BluetoothHeadsetReceiver.BTHeadsetPlugEvent) { - Timber.v("onBTHeadsetEvent $event") + Timber.tag(loggerTag.value).v("onBTHeadsetEvent $event") onAudioDeviceChange() } } diff --git a/vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserver.kt b/vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserver.kt index f14bb2f849..99c26c5ebe 100644 --- a/vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserver.kt +++ b/vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserver.kt @@ -16,6 +16,7 @@ package im.vector.app.features.call.webrtc +import org.matrix.android.sdk.api.logger.LoggerTag import org.matrix.android.sdk.api.session.call.CallState import org.matrix.android.sdk.api.session.call.MxPeerConnectionState import org.webrtc.DataChannel @@ -25,10 +26,12 @@ import org.webrtc.PeerConnection import org.webrtc.RtpReceiver import timber.log.Timber +private val loggerTag = LoggerTag("PeerConnectionObserver", LoggerTag.VOIP) + class PeerConnectionObserver(private val webRtcCall: WebRtcCall) : PeerConnection.Observer { override fun onConnectionChange(newState: PeerConnection.PeerConnectionState?) { - Timber.v("## VOIP StreamObserver onConnectionChange: $newState") + Timber.tag(loggerTag.value).v("StreamObserver onConnectionChange: $newState") when (newState) { /** * Every ICE transport used by the connection is either in use (state "connected" or "completed") @@ -79,20 +82,20 @@ class PeerConnectionObserver(private val webRtcCall: WebRtcCall) : PeerConnectio } override fun onIceCandidate(iceCandidate: IceCandidate) { - Timber.v("## VOIP StreamObserver onIceCandidate: $iceCandidate") + Timber.tag(loggerTag.value).v("StreamObserver onIceCandidate: $iceCandidate") webRtcCall.onIceCandidate(iceCandidate) } override fun onDataChannel(dc: DataChannel) { - Timber.v("## VOIP StreamObserver onDataChannel: ${dc.state()}") + Timber.tag(loggerTag.value).v("StreamObserver onDataChannel: ${dc.state()}") } override fun onIceConnectionReceivingChange(receiving: Boolean) { - Timber.v("## VOIP StreamObserver onIceConnectionReceivingChange: $receiving") + Timber.tag(loggerTag.value).v("StreamObserver onIceConnectionReceivingChange: $receiving") } override fun onIceConnectionChange(newState: PeerConnection.IceConnectionState) { - Timber.v("## VOIP StreamObserver onIceConnectionChange IceConnectionState:$newState") + Timber.tag(loggerTag.value).v("StreamObserver onIceConnectionChange IceConnectionState:$newState") when (newState) { /** * the ICE agent is gathering addresses or is waiting to be given remote candidates through @@ -145,29 +148,29 @@ class PeerConnectionObserver(private val webRtcCall: WebRtcCall) : PeerConnectio } override fun onAddStream(stream: MediaStream) { - Timber.v("## VOIP StreamObserver onAddStream: $stream") + Timber.tag(loggerTag.value).v("StreamObserver onAddStream: $stream") webRtcCall.onAddStream(stream) } override fun onRemoveStream(stream: MediaStream) { - Timber.v("## VOIP StreamObserver onRemoveStream") + Timber.tag(loggerTag.value).v("StreamObserver onRemoveStream") webRtcCall.onRemoveStream() } override fun onIceGatheringChange(newState: PeerConnection.IceGatheringState) { - Timber.v("## VOIP StreamObserver onIceGatheringChange: $newState") + Timber.tag(loggerTag.value).v("StreamObserver onIceGatheringChange: $newState") } override fun onSignalingChange(newState: PeerConnection.SignalingState) { - Timber.v("## VOIP StreamObserver onSignalingChange: $newState") + Timber.tag(loggerTag.value).v("StreamObserver onSignalingChange: $newState") } override fun onIceCandidatesRemoved(candidates: Array) { - Timber.v("## VOIP StreamObserver onIceCandidatesRemoved: ${candidates.contentToString()}") + Timber.tag(loggerTag.value).v("StreamObserver onIceCandidatesRemoved: ${candidates.contentToString()}") } override fun onRenegotiationNeeded() { - Timber.v("## VOIP StreamObserver onRenegotiationNeeded") + Timber.tag(loggerTag.value).v("StreamObserver onRenegotiationNeeded") webRtcCall.onRenegotiationNeeded(restartIce = false) } @@ -178,6 +181,6 @@ class PeerConnectionObserver(private val webRtcCall: WebRtcCall) : PeerConnectio * gets a new set of tracks because the media element being captured loaded a new source. */ override fun onAddTrack(p0: RtpReceiver?, p1: Array?) { - Timber.v("## VOIP StreamObserver onAddTrack") + Timber.tag(loggerTag.value).v("StreamObserver onAddTrack") } } diff --git a/vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserverAdapter.kt b/vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserverAdapter.kt deleted file mode 100644 index 3d31f0e705..0000000000 --- a/vector/src/main/java/im/vector/app/features/call/webrtc/PeerConnectionObserverAdapter.kt +++ /dev/null @@ -1,70 +0,0 @@ -/* - * Copyright (c) 2020 New Vector Ltd - * - * 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 im.vector.app.features.call.webrtc - -import org.webrtc.DataChannel -import org.webrtc.IceCandidate -import org.webrtc.MediaStream -import org.webrtc.PeerConnection -import org.webrtc.RtpReceiver -import timber.log.Timber - -abstract class PeerConnectionObserverAdapter : PeerConnection.Observer { - override fun onIceCandidate(p0: IceCandidate?) { - Timber.v("## VOIP onIceCandidate $p0") - } - - override fun onDataChannel(p0: DataChannel?) { - Timber.v("## VOIP onDataChannel $p0") - } - - override fun onIceConnectionReceivingChange(p0: Boolean) { - Timber.v("## VOIP onIceConnectionReceivingChange $p0") - } - - override fun onIceConnectionChange(p0: PeerConnection.IceConnectionState?) { - Timber.v("## VOIP onIceConnectionChange $p0") - } - - override fun onIceGatheringChange(p0: PeerConnection.IceGatheringState?) { - Timber.v("## VOIP onIceConnectionChange $p0") - } - - override fun onAddStream(mediaStream: MediaStream?) { - Timber.v("## VOIP onAddStream $mediaStream") - } - - override fun onSignalingChange(p0: PeerConnection.SignalingState?) { - Timber.v("## VOIP onSignalingChange $p0") - } - - override fun onIceCandidatesRemoved(p0: Array?) { - Timber.v("## VOIP onIceCandidatesRemoved $p0") - } - - override fun onRemoveStream(mediaStream: MediaStream?) { - Timber.v("## VOIP onRemoveStream $mediaStream") - } - - override fun onRenegotiationNeeded() { - Timber.v("## VOIP onRenegotiationNeeded") - } - - override fun onAddTrack(p0: RtpReceiver?, p1: Array?) { - Timber.v("## VOIP onAddTrack $p0 / out: $p1") - } -} diff --git a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt index 9bcd442be6..6c55b9380d 100644 --- a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt +++ b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCall.kt @@ -89,7 +89,7 @@ private const val AUDIO_TRACK_ID = "${STREAM_ID}a0" private const val VIDEO_TRACK_ID = "${STREAM_ID}v0" private val DEFAULT_AUDIO_CONSTRAINTS = MediaConstraints() -private val WebRtcCallTag = LoggerTag("WebRtcCall", LoggerTag.VOIP) +private val loggerTag = LoggerTag("WebRtcCall", LoggerTag.VOIP) class WebRtcCall( val mxCall: MxCall, @@ -195,7 +195,7 @@ class WebRtcCall( .subscribe { // omit empty :/ if (it.isNotEmpty()) { - Timber.tag(WebRtcCallTag.computedValue).v("Sending local ice candidates to call") + Timber.tag(loggerTag.value).v("Sending local ice candidates to call") // it.forEach { peerConnection?.addIceCandidate(it) } mxCall.sendLocalCallCandidates(it.mapToCallCandidate()) } @@ -213,7 +213,7 @@ class WebRtcCall( fun onRenegotiationNeeded(restartIce: Boolean) { sessionScope?.launch(dispatcher) { if (mxCall.state != CallState.CreateOffer && mxCall.opponentVersion == 0) { - Timber.tag(WebRtcCallTag.computedValue).v("Opponent does not support renegotiation: ignoring onRenegotiationNeeded event") + Timber.tag(loggerTag.value).v("Opponent does not support renegotiation: ignoring onRenegotiationNeeded event") return@launch } val constraints = MediaConstraints() @@ -221,7 +221,7 @@ class WebRtcCall( constraints.mandatory.add(MediaConstraints.KeyValuePair("IceRestart", "true")) } val peerConnection = peerConnection ?: return@launch - Timber.tag(WebRtcCallTag.computedValue).v("creating offer...") + Timber.tag(loggerTag.value).v("creating offer...") makingOffer = true try { val sessionDescription = peerConnection.awaitCreateOffer(constraints) ?: return@launch @@ -241,7 +241,7 @@ class WebRtcCall( } } catch (failure: Throwable) { // Need to handle error properly. - Timber.tag(WebRtcCallTag.computedValue).v("Failure while creating offer") + Timber.tag(loggerTag.value).v("Failure while creating offer") } finally { makingOffer = false } @@ -270,7 +270,7 @@ class WebRtcCall( } } } - Timber.tag(WebRtcCallTag.computedValue).v("creating peer connection...with iceServers $iceServers ") + Timber.tag(loggerTag.value).v("creating peer connection...with iceServers $iceServers ") val rtcConfig = PeerConnection.RTCConfiguration(iceServers).apply { sdpSemantics = PeerConnection.SdpSemantics.UNIFIED_PLAN } @@ -317,7 +317,7 @@ class WebRtcCall( fun acceptIncomingCall() { sessionScope?.launch { - Timber.tag(WebRtcCallTag.computedValue).v("acceptIncomingCall from state ${mxCall.state}") + Timber.tag(loggerTag.value).v("acceptIncomingCall from state ${mxCall.state}") if (mxCall.state == CallState.LocalRinging) { internalAcceptIncomingCall() } @@ -336,7 +336,7 @@ class WebRtcCall( sender.dtmf()?.insertDtmf(digit, 100, 70) return@launch } catch (failure: Throwable) { - Timber.tag(WebRtcCallTag.computedValue).v("Fail to send Dtmf digit") + Timber.tag(loggerTag.value).v("Fail to send Dtmf digit") } } } @@ -345,7 +345,7 @@ class WebRtcCall( fun attachViewRenderers(localViewRenderer: SurfaceViewRenderer?, remoteViewRenderer: SurfaceViewRenderer, mode: String?) { sessionScope?.launch(dispatcher) { - Timber.tag(WebRtcCallTag.computedValue).v("attachViewRenderers localRendeder $localViewRenderer / $remoteViewRenderer") + Timber.tag(loggerTag.value).v("attachViewRenderers localRendeder $localViewRenderer / $remoteViewRenderer") localSurfaceRenderers.addIfNeeded(localViewRenderer) remoteSurfaceRenderers.addIfNeeded(remoteViewRenderer) when (mode) { @@ -392,7 +392,7 @@ class WebRtcCall( } private suspend fun detachRenderersInternal(renderers: List?) = withContext(dispatcher) { - Timber.tag(WebRtcCallTag.computedValue).v("detachRenderers") + Timber.tag(loggerTag.value).v("detachRenderers") if (renderers.isNullOrEmpty()) { // remove all sinks localSurfaceRenderers.forEach { @@ -425,12 +425,12 @@ class WebRtcCall( // 2. Access camera (if video call) + microphone, create local stream createLocalStream() attachViewRenderersInternal() - Timber.tag(WebRtcCallTag.computedValue).v("remoteCandidateSource $remoteCandidateSource") + Timber.tag(loggerTag.value).v("remoteCandidateSource $remoteCandidateSource") remoteIceCandidateDisposable = remoteCandidateSource.subscribe({ - Timber.tag(WebRtcCallTag.computedValue).v("adding remote ice candidate $it") + Timber.tag(loggerTag.value).v("adding remote ice candidate $it") peerConnection?.addIceCandidate(it) }, { - Timber.tag(WebRtcCallTag.computedValue).v("failed to add remote ice candidate $it") + Timber.tag(loggerTag.value).v("failed to add remote ice candidate $it") }) // Now we wait for negotiation callback } @@ -456,14 +456,14 @@ class WebRtcCall( SessionDescription(SessionDescription.Type.OFFER, it) } if (offerSdp == null) { - Timber.tag(WebRtcCallTag.computedValue).v("We don't have any offer to process") + Timber.tag(loggerTag.value).v("We don't have any offer to process") return@withContext } - Timber.tag(WebRtcCallTag.computedValue).v("Offer sdp for invite: ${offerSdp.description}") + Timber.tag(loggerTag.value).v("Offer sdp for invite: ${offerSdp.description}") try { peerConnection?.awaitSetRemoteDescription(offerSdp) } catch (failure: Throwable) { - Timber.tag(WebRtcCallTag.computedValue).v("Failure putting remote description") + Timber.tag(loggerTag.value).v("Failure putting remote description") endCall(true, CallHangupContent.Reason.UNKWOWN_ERROR) return@withContext } @@ -475,12 +475,12 @@ class WebRtcCall( createAnswer()?.also { mxCall.accept(it.description) } - Timber.tag(WebRtcCallTag.computedValue).v("remoteCandidateSource $remoteCandidateSource") + Timber.tag(loggerTag.value).v("remoteCandidateSource $remoteCandidateSource") remoteIceCandidateDisposable = remoteCandidateSource.subscribe({ - Timber.tag(WebRtcCallTag.computedValue).v("adding remote ice candidate $it") + Timber.tag(loggerTag.value).v("adding remote ice candidate $it") peerConnection?.addIceCandidate(it) }, { - Timber.tag(WebRtcCallTag.computedValue).v("failed to add remote ice candidate $it") + Timber.tag(loggerTag.value).v("failed to add remote ice candidate $it") }) } @@ -492,7 +492,7 @@ class WebRtcCall( private fun createLocalStream() { val peerConnectionFactory = peerConnectionFactoryProvider.get() ?: return - Timber.tag(WebRtcCallTag.computedValue).v("Create local stream for call ${mxCall.callId}") + Timber.tag(loggerTag.value).v("Create local stream for call ${mxCall.callId}") configureAudioTrack(peerConnectionFactory) // add video track if needed if (mxCall.isVideoCall) { @@ -505,7 +505,7 @@ class WebRtcCall( val audioSource = peerConnectionFactory.createAudioSource(DEFAULT_AUDIO_CONSTRAINTS) val audioTrack = peerConnectionFactory.createAudioTrack(AUDIO_TRACK_ID, audioSource) audioTrack.setEnabled(true) - Timber.tag(WebRtcCallTag.computedValue).v("Add audio track $AUDIO_TRACK_ID to call ${mxCall.callId}") + Timber.tag(loggerTag.value).v("Add audio track $AUDIO_TRACK_ID to call ${mxCall.callId}") peerConnection?.addTrack(audioTrack, listOf(STREAM_ID)) localAudioSource = audioSource localAudioTrack = audioTrack @@ -547,7 +547,7 @@ class WebRtcCall( override fun onCameraClosed() { super.onCameraClosed() - Timber.tag(WebRtcCallTag.computedValue).v("onCameraClosed") + Timber.tag(loggerTag.value).v("onCameraClosed") // This could happen if you open the camera app in chat // We then register in order to restart capture as soon as the camera is available again videoCapturerIsInError = true @@ -555,16 +555,16 @@ class WebRtcCall( cameraAvailabilityCallback = object : CameraManager.AvailabilityCallback() { override fun onCameraUnavailable(cameraId: String) { super.onCameraUnavailable(cameraId) - Timber.tag(WebRtcCallTag.computedValue).v("On camera unavailable: $cameraId") + Timber.tag(loggerTag.value).v("On camera unavailable: $cameraId") } override fun onCameraAccessPrioritiesChanged() { super.onCameraAccessPrioritiesChanged() - Timber.tag(WebRtcCallTag.computedValue).v("onCameraAccessPrioritiesChanged") + Timber.tag(loggerTag.value).v("onCameraAccessPrioritiesChanged") } override fun onCameraAvailable(cameraId: String) { - Timber.tag(WebRtcCallTag.computedValue).v("On camera available: $cameraId") + Timber.tag(loggerTag.value).v("On camera available: $cameraId") if (cameraId == camera.name) { videoCapturer?.startCapture(currentCaptureFormat.width, currentCaptureFormat.height, currentCaptureFormat.fps) cameraManager?.unregisterAvailabilityCallback(this) @@ -577,7 +577,7 @@ class WebRtcCall( val videoSource = peerConnectionFactory.createVideoSource(videoCapturer.isScreencast) val surfaceTextureHelper = SurfaceTextureHelper.create("CaptureThread", rootEglBase!!.eglBaseContext) - Timber.tag(WebRtcCallTag.computedValue).v("Local video source created") + Timber.tag(loggerTag.value).v("Local video source created") videoCapturer.initialize(surfaceTextureHelper, context, videoSource.capturerObserver) // HD @@ -585,7 +585,7 @@ class WebRtcCall( this.videoCapturer = videoCapturer val videoTrack = peerConnectionFactory.createVideoTrack(VIDEO_TRACK_ID, videoSource) - Timber.tag(WebRtcCallTag.computedValue).v("Add video track $VIDEO_TRACK_ID to call ${mxCall.callId}") + Timber.tag(loggerTag.value).v("Add video track $VIDEO_TRACK_ID to call ${mxCall.callId}") videoTrack.setEnabled(true) peerConnection?.addTrack(videoTrack, listOf(STREAM_ID)) localVideoSource = videoSource @@ -595,7 +595,7 @@ class WebRtcCall( fun setCaptureFormat(format: CaptureFormat) { sessionScope?.launch(dispatcher) { - Timber.tag(WebRtcCallTag.computedValue).v("setCaptureFormat $format") + Timber.tag(loggerTag.value).v("setCaptureFormat $format") videoCapturer?.changeCaptureFormat(format.width, format.height, format.fps) currentCaptureFormat = format } @@ -689,14 +689,14 @@ class WebRtcCall( fun switchCamera() { sessionScope?.launch(dispatcher) { - Timber.tag(WebRtcCallTag.computedValue).v("switchCamera") + Timber.tag(loggerTag.value).v("switchCamera") if (mxCall.state is CallState.Connected && mxCall.isVideoCall) { val oppositeCamera = getOppositeCameraIfAny() ?: return@launch videoCapturer?.switchCamera( object : CameraVideoCapturer.CameraSwitchHandler { // Invoked on success. |isFrontCamera| is true if the new camera is front facing. override fun onCameraSwitchDone(isFrontCamera: Boolean) { - Timber.tag(WebRtcCallTag.computedValue).v("onCameraSwitchDone isFront $isFrontCamera") + Timber.tag(loggerTag.value).v("onCameraSwitchDone isFront $isFrontCamera") cameraInUse = oppositeCamera localSurfaceRenderers.forEach { it.get()?.setMirror(isFrontCamera) @@ -707,7 +707,7 @@ class WebRtcCall( } override fun onCameraSwitchError(errorDescription: String?) { - Timber.tag(WebRtcCallTag.computedValue).v("onCameraSwitchError isFront $errorDescription") + Timber.tag(loggerTag.value).v("onCameraSwitchError isFront $errorDescription") } }, oppositeCamera.name ) @@ -716,7 +716,7 @@ class WebRtcCall( } private suspend fun createAnswer(): SessionDescription? { - Timber.tag(WebRtcCallTag.computedValue).w("createAnswer") + Timber.tag(loggerTag.value).w("createAnswer") val peerConnection = peerConnection ?: return null val constraints = MediaConstraints().apply { mandatory.add(MediaConstraints.KeyValuePair("OfferToReceiveAudio", "true")) @@ -727,7 +727,7 @@ class WebRtcCall( peerConnection.awaitSetLocalDescription(localDescription) localDescription } catch (failure: Throwable) { - Timber.tag(WebRtcCallTag.computedValue).v("Fail to create answer") + Timber.tag(loggerTag.value).v("Fail to create answer") null } } @@ -768,7 +768,7 @@ class WebRtcCall( sessionScope?.launch(dispatcher) { // reportError("Weird-looking stream: " + stream); if (stream.audioTracks.size > 1 || stream.videoTracks.size > 1) { - Timber.tag(WebRtcCallTag.computedValue).e("## VOIP StreamObserver weird looking stream: $stream") + Timber.tag(loggerTag.value).e("StreamObserver weird looking stream: $stream") // TODO maybe do something more?? endCall(true) return@launch @@ -832,7 +832,7 @@ class WebRtcCall( if (it.sdpMid.isNullOrEmpty() || it.candidate.isNullOrEmpty()) { return@forEach } - Timber.tag(WebRtcCallTag.computedValue).v("onCallIceCandidateReceived for call ${mxCall.callId} sdp: ${it.candidate}") + Timber.tag(loggerTag.value).v("onCallIceCandidateReceived for call ${mxCall.callId} sdp: ${it.candidate}") val iceCandidate = IceCandidate(it.sdpMid, it.sdpMLineIndex, it.candidate) remoteCandidateSource.onNext(iceCandidate) } @@ -841,7 +841,7 @@ class WebRtcCall( fun onCallAnswerReceived(callAnswerContent: CallAnswerContent) { sessionScope?.launch(dispatcher) { - Timber.tag(WebRtcCallTag.computedValue).v("onCallAnswerReceived ${callAnswerContent.callId}") + Timber.tag(loggerTag.value).v("onCallAnswerReceived ${callAnswerContent.callId}") val sdp = SessionDescription(SessionDescription.Type.ANSWER, callAnswerContent.answer.sdp) try { peerConnection?.awaitSetRemoteDescription(sdp) @@ -861,7 +861,7 @@ class WebRtcCall( val type = description?.type val sdpText = description?.sdp if (type == null || sdpText == null) { - Timber.tag(WebRtcCallTag.computedValue).i("Ignoring invalid m.call.negotiate event") + Timber.tag(loggerTag.value).i("Ignoring invalid m.call.negotiate event") return@launch } val peerConnection = peerConnection ?: return@launch @@ -876,7 +876,7 @@ class WebRtcCall( ignoreOffer = !polite && offerCollision if (ignoreOffer) { - Timber.tag(WebRtcCallTag.computedValue).i("Ignoring colliding negotiate event because we're impolite") + Timber.tag(loggerTag.value).i("Ignoring colliding negotiate event because we're impolite") return@launch } val prevOnHold = computeIsLocalOnHold() @@ -889,7 +889,7 @@ class WebRtcCall( } } } catch (failure: Throwable) { - Timber.tag(WebRtcCallTag.computedValue).e(failure, "Failed to complete negotiation") + Timber.tag(loggerTag.value).e(failure, "Failed to complete negotiation") } val nowOnHold = computeIsLocalOnHold() wasLocalOnHold = nowOnHold @@ -912,7 +912,7 @@ class WebRtcCall( val session = sessionProvider.get() ?: return@launch val newAssertedIdentity = callAssertedIdentityContent.assertedIdentity ?: return@launch if (newAssertedIdentity.id == null && newAssertedIdentity.displayName == null) { - Timber.tag(WebRtcCallTag.computedValue).v("Asserted identity received with no relevant information, skip") + Timber.tag(loggerTag.value).v("Asserted identity received with no relevant information, skip") return@launch } remoteAssertedIdentity = newAssertedIdentity diff --git a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt index 566ba7e81a..1d09d30de2 100644 --- a/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt +++ b/vector/src/main/java/im/vector/app/features/call/webrtc/WebRtcCallManager.kt @@ -61,7 +61,7 @@ import javax.inject.Singleton * Manage peerConnectionFactory & Peer connections outside of activity lifecycle to resist configuration changes * Use app context */ -private val WebRtcCallManagerTag = LoggerTag("WebRtcCallManager", LoggerTag.VOIP) +private val loggerTag = LoggerTag("WebRtcCallManager", LoggerTag.VOIP) @Singleton class WebRtcCallManager @Inject constructor( @@ -187,7 +187,7 @@ class WebRtcCallManager @Inject constructor( fun getAdvertisedCalls() = advertisedCalls fun headSetButtonTapped() { - Timber.tag(WebRtcCallManagerTag.computedValue).v("headSetButtonTapped") + Timber.tag(loggerTag.value).v("headSetButtonTapped") val call = getCurrentCall() ?: return if (call.mxCall.state is CallState.LocalRinging) { call.acceptIncomingCall() @@ -200,12 +200,12 @@ class WebRtcCallManager @Inject constructor( private fun createPeerConnectionFactoryIfNeeded() { if (peerConnectionFactory != null) return - Timber.tag(WebRtcCallManagerTag.computedValue).v("createPeerConnectionFactory") + Timber.tag(loggerTag.value).v("createPeerConnectionFactory") val eglBaseContext = rootEglBase?.eglBaseContext ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).e("No EGL BASE") + Timber.tag(loggerTag.value).e("No EGL BASE") } - Timber.tag(WebRtcCallManagerTag.computedValue).v("PeerConnectionFactory.initialize") + Timber.tag(loggerTag.value).v("PeerConnectionFactory.initialize") PeerConnectionFactory.initialize(PeerConnectionFactory .InitializationOptions.builder(context.applicationContext) .createInitializationOptions() @@ -219,7 +219,7 @@ class WebRtcCallManager @Inject constructor( /* enableH264HighProfile */ true) val defaultVideoDecoderFactory = DefaultVideoDecoderFactory(eglBaseContext) - Timber.tag(WebRtcCallManagerTag.computedValue).v("PeerConnectionFactory.createPeerConnectionFactory ...") + Timber.tag(loggerTag.value).v("PeerConnectionFactory.createPeerConnectionFactory ...") peerConnectionFactory = PeerConnectionFactory.builder() .setOptions(options) .setVideoEncoderFactory(defaultVideoEncoderFactory) @@ -228,7 +228,7 @@ class WebRtcCallManager @Inject constructor( } private fun onCallActive(call: WebRtcCall) { - Timber.tag(WebRtcCallManagerTag.computedValue).v("WebRtcPeerConnectionManager onCall active: ${call.mxCall.callId}") + Timber.tag(loggerTag.value).v("WebRtcPeerConnectionManager onCall active: ${call.mxCall.callId}") val currentCall = getCurrentCall().takeIf { it != call } currentCall?.updateRemoteOnHold(onHold = true) audioManager.setMode(if (call.mxCall.isVideoCall) CallAudioManager.Mode.VIDEO_CALL else CallAudioManager.Mode.AUDIO_CALL) @@ -236,9 +236,9 @@ class WebRtcCallManager @Inject constructor( } private fun onCallEnded(callId: String) { - Timber.tag(WebRtcCallManagerTag.computedValue).v("WebRtcPeerConnectionManager onCall ended: $callId") + Timber.tag(loggerTag.value).v("WebRtcPeerConnectionManager onCall ended: $callId") val webRtcCall = callsByCallId.remove(callId) ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).v("On call ended for unknown call $callId") + Timber.tag(loggerTag.value).v("On call ended for unknown call $callId") } CallService.onCallTerminated(context, callId) callsByRoomId[webRtcCall.signalingRoomId]?.remove(webRtcCall) @@ -250,7 +250,7 @@ class WebRtcCallManager @Inject constructor( } // There is no active calls if (getCurrentCall() == null) { - Timber.tag(WebRtcCallManagerTag.computedValue).v("Dispose peerConnectionFactory as there is no need to keep one") + Timber.tag(loggerTag.value).v("Dispose peerConnectionFactory as there is no need to keep one") peerConnectionFactory?.dispose() peerConnectionFactory = null audioManager.setMode(CallAudioManager.Mode.DEFAULT) @@ -268,13 +268,13 @@ class WebRtcCallManager @Inject constructor( suspend fun startOutgoingCall(nativeRoomId: String, otherUserId: String, isVideoCall: Boolean, transferee: WebRtcCall? = null) { val signalingRoomId = callUserMapper?.getOrCreateVirtualRoomForRoom(nativeRoomId, otherUserId) ?: nativeRoomId - Timber.tag(WebRtcCallManagerTag.computedValue).v("startOutgoingCall in room $signalingRoomId to $otherUserId isVideo $isVideoCall") + Timber.tag(loggerTag.value).v("startOutgoingCall in room $signalingRoomId to $otherUserId isVideo $isVideoCall") if (getCallsByRoomId(nativeRoomId).isNotEmpty()) { - Timber.tag(WebRtcCallManagerTag.computedValue).w("you already have a call in this room") + Timber.tag(loggerTag.value).w("you already have a call in this room") return } if (getCurrentCall() != null && getCurrentCall()?.mxCall?.state !is CallState.Connected || getCalls().size >= 2) { - Timber.tag(WebRtcCallManagerTag.computedValue).w("cannot start outgoing call") + Timber.tag(loggerTag.value).w("cannot start outgoing call") // Just ignore, maybe we could answer from other session? return } @@ -297,10 +297,10 @@ class WebRtcCallManager @Inject constructor( } override fun onCallIceCandidateReceived(mxCall: MxCall, iceCandidatesContent: CallCandidatesContent) { - Timber.tag(WebRtcCallManagerTag.computedValue).v("onCallIceCandidateReceived for call ${mxCall.callId}") + Timber.tag(loggerTag.value).v("onCallIceCandidateReceived for call ${mxCall.callId}") val call = callsByCallId[iceCandidatesContent.callId] ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallIceCandidateReceived for non active call? ${iceCandidatesContent.callId}") + Timber.tag(loggerTag.value).w("onCallIceCandidateReceived for non active call? ${iceCandidatesContent.callId}") } call.onCallIceCandidateReceived(iceCandidatesContent) } @@ -337,14 +337,14 @@ class WebRtcCallManager @Inject constructor( } override fun onCallInviteReceived(mxCall: MxCall, callInviteContent: CallInviteContent) { - Timber.tag(WebRtcCallManagerTag.computedValue).v("onCallInviteReceived callId ${mxCall.callId}") + Timber.tag(loggerTag.value).v("onCallInviteReceived callId ${mxCall.callId}") val nativeRoomId = callUserMapper?.nativeRoomForVirtualRoom(mxCall.roomId) ?: mxCall.roomId if (getCallsByRoomId(nativeRoomId).isNotEmpty()) { - Timber.tag(WebRtcCallManagerTag.computedValue).w("you already have a call in this room") + Timber.tag(loggerTag.value).w("you already have a call in this room") return } if ((getCurrentCall() != null && getCurrentCall()?.mxCall?.state !is CallState.Connected) || getCalls().size >= 2) { - Timber.tag(WebRtcCallManagerTag.computedValue).w("receiving incoming call but cannot handle it") + Timber.tag(loggerTag.value).w("receiving incoming call but cannot handle it") // Just ignore, maybe we could answer from other session? return } @@ -373,7 +373,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallAnswerReceived(callAnswerContent: CallAnswerContent) { val call = callsByCallId[callAnswerContent.callId] ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallAnswerReceived for non active call? ${callAnswerContent.callId}") + Timber.tag(loggerTag.value).w("onCallAnswerReceived for non active call? ${callAnswerContent.callId}") } val mxCall = call.mxCall // Update service state @@ -387,7 +387,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallHangupReceived(callHangupContent: CallHangupContent) { val call = callsByCallId[callHangupContent.callId] ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallHangupReceived for non active call? ${callHangupContent.callId}") + Timber.tag(loggerTag.value).w("onCallHangupReceived for non active call? ${callHangupContent.callId}") } call.endCall(false) } @@ -395,7 +395,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallRejectReceived(callRejectContent: CallRejectContent) { val call = callsByCallId[callRejectContent.callId] ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallRejectReceived for non active call? ${callRejectContent.callId}") + Timber.tag(loggerTag.value).w("onCallRejectReceived for non active call? ${callRejectContent.callId}") } call.endCall(false) } @@ -403,7 +403,7 @@ class WebRtcCallManager @Inject constructor( override fun onCallSelectAnswerReceived(callSelectAnswerContent: CallSelectAnswerContent) { val call = callsByCallId[callSelectAnswerContent.callId] ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallSelectAnswerReceived for non active call? ${callSelectAnswerContent.callId}") + Timber.tag(loggerTag.value).w("onCallSelectAnswerReceived for non active call? ${callSelectAnswerContent.callId}") } val selectedPartyId = callSelectAnswerContent.selectedPartyId if (selectedPartyId != call.mxCall.ourPartyId) { @@ -416,13 +416,13 @@ class WebRtcCallManager @Inject constructor( override fun onCallNegotiateReceived(callNegotiateContent: CallNegotiateContent) { val call = callsByCallId[callNegotiateContent.callId] ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallNegotiateReceived for non active call? ${callNegotiateContent.callId}") + Timber.tag(loggerTag.value).w("onCallNegotiateReceived for non active call? ${callNegotiateContent.callId}") } call.onCallNegotiateReceived(callNegotiateContent) } override fun onCallManagedByOtherSession(callId: String) { - Timber.tag(WebRtcCallManagerTag.computedValue).v("onCallManagedByOtherSession: $callId") + Timber.tag(loggerTag.value).v("onCallManagedByOtherSession: $callId") onCallEnded(callId) } @@ -432,7 +432,7 @@ class WebRtcCallManager @Inject constructor( } val call = callsByCallId[callAssertedIdentityContent.callId] ?: return Unit.also { - Timber.tag(WebRtcCallManagerTag.computedValue).w("onCallAssertedIdentityReceived for non active call? ${callAssertedIdentityContent.callId}") + Timber.tag(loggerTag.value).w("onCallAssertedIdentityReceived for non active call? ${callAssertedIdentityContent.callId}") } call.onCallAssertedIdentityReceived(callAssertedIdentityContent) } From e3464f58fa467b00a9a2e4a57e3989b644e84686 Mon Sep 17 00:00:00 2001 From: ganfra Date: Thu, 22 Jul 2021 15:39:13 +0200 Subject: [PATCH 3/4] Call notif: make end call notif better --- .../java/im/vector/app/core/services/CallService.kt | 10 ++++------ .../app/features/notifications/NotificationUtils.kt | 3 ++- 2 files changed, 6 insertions(+), 7 deletions(-) diff --git a/vector/src/main/java/im/vector/app/core/services/CallService.kt b/vector/src/main/java/im/vector/app/core/services/CallService.kt index 916dc66718..d8cf8cf6b8 100644 --- a/vector/src/main/java/im/vector/app/core/services/CallService.kt +++ b/vector/src/main/java/im/vector/app/core/services/CallService.kt @@ -210,13 +210,11 @@ class CallService : VectorService() { myStopSelf() } val wasConnected = connectedCallIds.remove(callId) + val notification = notificationUtils.buildCallEndedNotification(terminatedCall.isVideoCall) + notificationManager.notify(callId.hashCode(), notification) if (!wasConnected && !terminatedCall.isOutgoing && !rejected && endCallReason != EndCallReason.ANSWERED_ELSEWHERE) { - val notification = notificationUtils.buildCallMissedNotification(terminatedCall) - notificationManager.cancel(callId.hashCode()) - notificationManager.notify(MISSED_CALL_TAG, terminatedCall.nativeRoomId.hashCode(), notification) - } else { - val notification = notificationUtils.buildCallEndedNotification(terminatedCall.isVideoCall) - notificationManager.notify(callId.hashCode(), notification) + val missedCallNotification = notificationUtils.buildCallMissedNotification(terminatedCall) + notificationManager.notify(MISSED_CALL_TAG, terminatedCall.nativeRoomId.hashCode(), missedCallNotification) } } diff --git a/vector/src/main/java/im/vector/app/features/notifications/NotificationUtils.kt b/vector/src/main/java/im/vector/app/features/notifications/NotificationUtils.kt index 24a64f3cdb..cf13150e59 100755 --- a/vector/src/main/java/im/vector/app/features/notifications/NotificationUtils.kt +++ b/vector/src/main/java/im/vector/app/features/notifications/NotificationUtils.kt @@ -468,7 +468,8 @@ class NotificationUtils @Inject constructor(private val context: Context, setSmallIcon(R.drawable.ic_call_answer) } } - .setTimeoutAfter(2000) + // This is a trick to make the previous notification with same id disappear as cancel notification is not working with Foreground Service. + .setTimeoutAfter(1) .setColor(ThemeUtils.getColor(context, android.R.attr.colorPrimary)) .setCategory(NotificationCompat.CATEGORY_CALL) .build() From 45a51d138e88070360faa5d76f7c6a2b526e8563 Mon Sep 17 00:00:00 2001 From: ganfra Date: Fri, 23 Jul 2021 09:50:09 +0200 Subject: [PATCH 4/4] Voip log: clean after Benoit review --- .../matrix/android/sdk/api/logger/LoggerTag.kt | 15 ++++++--------- .../internal/session/call/CallSignalingHandler.kt | 3 ++- 2 files changed, 8 insertions(+), 10 deletions(-) diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt index 9978ae32aa..9e7ee297f1 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/api/logger/LoggerTag.kt @@ -22,16 +22,13 @@ package org.matrix.android.sdk.api.logger * val loggerTag = LoggerTag("MyTag", LoggerTag.VOIP) * Timber.tag(loggerTag.value).v("My log message") */ -open class LoggerTag(private val _value: String, private val parentTag: LoggerTag? = null) { +open class LoggerTag(_value: String, parentTag: LoggerTag? = null) { object VOIP : LoggerTag("VOIP", null) - val value: String - get() { - return if (parentTag == null) { - _value - } else { - "${parentTag.value}/$_value" - } - } + val value: String = if (parentTag == null) { + _value + } else { + "${parentTag.value}/$_value" + } } diff --git a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt index a71fdeae03..59058bf976 100644 --- a/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt +++ b/matrix-sdk-android/src/main/java/org/matrix/android/sdk/internal/session/call/CallSignalingHandler.kt @@ -38,6 +38,7 @@ import timber.log.Timber import javax.inject.Inject private val loggerTag = LoggerTag("CallSignalingHandler", LoggerTag.VOIP) +private const val MAX_AGE_TO_RING = 40_000 @SessionScope internal class CallSignalingHandler @Inject constructor(private val activeCallHandler: ActiveCallHandler, @@ -185,7 +186,7 @@ internal class CallSignalingHandler @Inject constructor(private val activeCallHa } val now = System.currentTimeMillis() val age = now - (event.ageLocalTs ?: now) - if (age > 40_000 && event.getClearType() == EventType.CALL_INVITE) { + if (age > MAX_AGE_TO_RING) { Timber.tag(loggerTag.value).w("Call invite is too old to ring.") return }