Use SDK to get Element Wellknown content.
This commit is contained in:
parent
6531a39b5a
commit
8ec283f2ca
10 changed files with 231 additions and 88 deletions
|
|
@ -20,6 +20,7 @@ import com.squareup.anvil.annotations.ContributesBinding
|
||||||
import io.element.android.appconfig.ElementCallConfig
|
import io.element.android.appconfig.ElementCallConfig
|
||||||
import io.element.android.libraries.di.AppScope
|
import io.element.android.libraries.di.AppScope
|
||||||
import io.element.android.libraries.matrix.api.MatrixClientProvider
|
import io.element.android.libraries.matrix.api.MatrixClientProvider
|
||||||
|
import io.element.android.libraries.matrix.api.call.ElementCallBaseUrlProvider
|
||||||
import io.element.android.libraries.matrix.api.core.RoomId
|
import io.element.android.libraries.matrix.api.core.RoomId
|
||||||
import io.element.android.libraries.matrix.api.core.SessionId
|
import io.element.android.libraries.matrix.api.core.SessionId
|
||||||
import io.element.android.libraries.matrix.api.widget.CallWidgetSettingsProvider
|
import io.element.android.libraries.matrix.api.widget.CallWidgetSettingsProvider
|
||||||
|
|
@ -41,9 +42,10 @@ class DefaultCallWidgetProvider @Inject constructor(
|
||||||
languageTag: String?,
|
languageTag: String?,
|
||||||
theme: String?,
|
theme: String?,
|
||||||
): Result<CallWidgetProvider.GetWidgetResult> = runCatching {
|
): Result<CallWidgetProvider.GetWidgetResult> = runCatching {
|
||||||
val room = matrixClientsProvider.getOrRestore(sessionId).getOrThrow().getRoom(roomId) ?: error("Room not found")
|
val matrixClient = matrixClientsProvider.getOrRestore(sessionId).getOrThrow()
|
||||||
|
val room = matrixClient.getRoom(roomId) ?: error("Room not found")
|
||||||
val baseUrl = appPreferencesStore.getCustomElementCallBaseUrlFlow().firstOrNull()
|
val baseUrl = appPreferencesStore.getCustomElementCallBaseUrlFlow().firstOrNull()
|
||||||
?: elementCallBaseUrlProvider.provides(sessionId)
|
?: elementCallBaseUrlProvider.provides(matrixClient)
|
||||||
?: ElementCallConfig.DEFAULT_BASE_URL
|
?: ElementCallConfig.DEFAULT_BASE_URL
|
||||||
val widgetSettings = callWidgetSettingsProvider.provide(baseUrl, encrypted = room.isEncrypted)
|
val widgetSettings = callWidgetSettingsProvider.provide(baseUrl, encrypted = room.isEncrypted)
|
||||||
val callUrl = room.generateWidgetWebViewUrl(
|
val callUrl = room.generateWidgetWebViewUrl(
|
||||||
|
|
|
||||||
|
|
@ -1,61 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (c) 2024 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 io.element.android.features.call.impl.utils
|
|
||||||
|
|
||||||
import com.squareup.anvil.annotations.ContributesBinding
|
|
||||||
import io.element.android.features.call.impl.wellknown.CallWellknownAPI
|
|
||||||
import io.element.android.libraries.core.coroutine.CoroutineDispatchers
|
|
||||||
import io.element.android.libraries.di.AppScope
|
|
||||||
import io.element.android.libraries.di.SingleIn
|
|
||||||
import io.element.android.libraries.matrix.api.core.SessionId
|
|
||||||
import io.element.android.libraries.network.RetrofitFactory
|
|
||||||
import kotlinx.coroutines.withContext
|
|
||||||
import retrofit2.HttpException
|
|
||||||
import timber.log.Timber
|
|
||||||
import java.net.HttpURLConnection
|
|
||||||
import javax.inject.Inject
|
|
||||||
|
|
||||||
interface ElementCallBaseUrlProvider {
|
|
||||||
suspend fun provides(sessionId: SessionId): String?
|
|
||||||
}
|
|
||||||
|
|
||||||
@SingleIn(AppScope::class)
|
|
||||||
@ContributesBinding(AppScope::class)
|
|
||||||
class DefaultElementCallBaseUrlProvider @Inject constructor(
|
|
||||||
private val retrofitFactory: RetrofitFactory,
|
|
||||||
private val coroutineDispatchers: CoroutineDispatchers,
|
|
||||||
) : ElementCallBaseUrlProvider {
|
|
||||||
private val apiCache = mutableMapOf<SessionId, CallWellknownAPI>()
|
|
||||||
|
|
||||||
override suspend fun provides(sessionId: SessionId): String? = withContext(coroutineDispatchers.io) {
|
|
||||||
val domain = sessionId.value.substringAfter(":")
|
|
||||||
val callWellknownAPI = apiCache.getOrPut(sessionId) {
|
|
||||||
retrofitFactory.create("https://$domain")
|
|
||||||
.create(CallWellknownAPI::class.java)
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
callWellknownAPI.getCallWellKnown().widgetUrl
|
|
||||||
} catch (e: HttpException) {
|
|
||||||
// Ignore Http 404, but re-throws any other exceptions
|
|
||||||
if (e.code() != HttpURLConnection.HTTP_NOT_FOUND) {
|
|
||||||
throw e
|
|
||||||
}
|
|
||||||
Timber.w(e, "Failed to fetch wellknown data")
|
|
||||||
null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -127,4 +127,9 @@ interface MatrixClient : Closeable {
|
||||||
* compute it manually.
|
* compute it manually.
|
||||||
*/
|
*/
|
||||||
fun userIdServerName(): String
|
fun userIdServerName(): String
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute generic GET requests through the SDKs internal HTTP client.
|
||||||
|
*/
|
||||||
|
suspend fun getUrl(url: String): Result<String>
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -5,7 +5,7 @@
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
* You may obtain a copy of the License at
|
* You may obtain a copy of the License at
|
||||||
*
|
*
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
*
|
*
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
|
@ -14,11 +14,10 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package io.element.android.features.call.impl.wellknown
|
package io.element.android.libraries.matrix.api.call
|
||||||
|
|
||||||
import retrofit2.http.GET
|
import io.element.android.libraries.matrix.api.MatrixClient
|
||||||
|
|
||||||
internal interface CallWellknownAPI {
|
interface ElementCallBaseUrlProvider {
|
||||||
@GET(".well-known/element/call.json")
|
suspend fun provides(matrixClient: MatrixClient): String?
|
||||||
suspend fun getCallWellKnown(): CallWellKnown
|
|
||||||
}
|
}
|
||||||
|
|
@ -291,6 +291,12 @@ class RustMatrixClient(
|
||||||
?: sessionId.value.substringAfter(":")
|
?: sessionId.value.substringAfter(":")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override suspend fun getUrl(url: String): Result<String> = withContext(sessionDispatcher) {
|
||||||
|
runCatching {
|
||||||
|
client.getUrl(url)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
override suspend fun getRoom(roomId: RoomId): MatrixRoom? {
|
override suspend fun getRoom(roomId: RoomId): MatrixRoom? {
|
||||||
return roomFactory.create(roomId)
|
return roomFactory.create(roomId)
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,52 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 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
|
||||||
|
*
|
||||||
|
* https://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 io.element.android.libraries.matrix.impl.call
|
||||||
|
|
||||||
|
import com.squareup.anvil.annotations.ContributesBinding
|
||||||
|
import io.element.android.libraries.di.AppScope
|
||||||
|
import io.element.android.libraries.matrix.api.MatrixClient
|
||||||
|
import io.element.android.libraries.matrix.api.call.ElementCallBaseUrlProvider
|
||||||
|
import timber.log.Timber
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
@ContributesBinding(AppScope::class)
|
||||||
|
class DefaultElementCallBaseUrlProvider @Inject constructor(
|
||||||
|
private val elementWellKnownParser: ElementWellKnownParser,
|
||||||
|
) : ElementCallBaseUrlProvider {
|
||||||
|
override suspend fun provides(matrixClient: MatrixClient): String? {
|
||||||
|
val url = buildString {
|
||||||
|
append("https://")
|
||||||
|
append(matrixClient.userIdServerName())
|
||||||
|
append("/.well-known/element/element.json")
|
||||||
|
}
|
||||||
|
return matrixClient.getUrl(url)
|
||||||
|
.onFailure { failure ->
|
||||||
|
Timber.w(failure, "Failed to fetch well-known element.json")
|
||||||
|
}
|
||||||
|
.getOrNull()
|
||||||
|
?.let { wellKnownStr ->
|
||||||
|
elementWellKnownParser.parse(wellKnownStr)
|
||||||
|
.onFailure { failure ->
|
||||||
|
// Can be a HTML 404.
|
||||||
|
Timber.w(failure, "Failed to parse content")
|
||||||
|
}
|
||||||
|
.getOrNull()
|
||||||
|
}
|
||||||
|
?.call
|
||||||
|
?.widgetUrl
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,35 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 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
|
||||||
|
*
|
||||||
|
* https://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 io.element.android.libraries.matrix.impl.call
|
||||||
|
|
||||||
|
import com.squareup.anvil.annotations.ContributesBinding
|
||||||
|
import io.element.android.libraries.di.AppScope
|
||||||
|
import org.matrix.rustcomponents.sdk.ElementWellKnown
|
||||||
|
import org.matrix.rustcomponents.sdk.makeElementWellKnown
|
||||||
|
|
||||||
|
interface ElementWellKnownParser {
|
||||||
|
fun parse(str: String): Result<ElementWellKnown>
|
||||||
|
}
|
||||||
|
|
||||||
|
@ContributesBinding(AppScope::class)
|
||||||
|
class RustElementWellKnownParser : ElementWellKnownParser {
|
||||||
|
override fun parse(str: String): Result<ElementWellKnown> {
|
||||||
|
return runCatching {
|
||||||
|
makeElementWellKnown(str)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,107 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2024 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
|
||||||
|
*
|
||||||
|
* https://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 io.element.android.libraries.matrix.impl.call
|
||||||
|
|
||||||
|
import com.google.common.truth.Truth.assertThat
|
||||||
|
import io.element.android.libraries.matrix.test.AN_EXCEPTION
|
||||||
|
import io.element.android.libraries.matrix.test.FakeMatrixClient
|
||||||
|
import io.element.android.tests.testutils.lambda.lambdaRecorder
|
||||||
|
import io.element.android.tests.testutils.lambda.value
|
||||||
|
import kotlinx.coroutines.test.runTest
|
||||||
|
import org.junit.Test
|
||||||
|
import org.matrix.rustcomponents.sdk.ElementCallWellKnown
|
||||||
|
import org.matrix.rustcomponents.sdk.ElementWellKnown
|
||||||
|
|
||||||
|
class DefaultElementCallBaseUrlProviderTest {
|
||||||
|
@Test
|
||||||
|
fun `provides returns null when getUrl returns an error`() = runTest {
|
||||||
|
val userIdServerNameLambda = lambdaRecorder<String> { "example.com" }
|
||||||
|
val getUrlLambda = lambdaRecorder<String, Result<String>> { _ ->
|
||||||
|
Result.failure(AN_EXCEPTION)
|
||||||
|
}
|
||||||
|
val sut = DefaultElementCallBaseUrlProvider(
|
||||||
|
FakeElementWellKnownParser(
|
||||||
|
Result.success(createElementWellKnown(""))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
val matrixClient = FakeMatrixClient(
|
||||||
|
userIdServerNameLambda = userIdServerNameLambda,
|
||||||
|
getUrlLambda = getUrlLambda,
|
||||||
|
)
|
||||||
|
val result = sut.provides(matrixClient)
|
||||||
|
assertThat(result).isNull()
|
||||||
|
userIdServerNameLambda.assertions().isCalledOnce()
|
||||||
|
getUrlLambda.assertions().isCalledOnce()
|
||||||
|
.with(value("https://example.com/.well-known/element/element.json"))
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `provides returns null when content parsing fails`() = runTest {
|
||||||
|
val userIdServerNameLambda = lambdaRecorder<String> { "example.com" }
|
||||||
|
val getUrlLambda = lambdaRecorder<String, Result<String>> { _ ->
|
||||||
|
Result.success("""{"call":{"widget_url":"https://example.com/call"}}""")
|
||||||
|
}
|
||||||
|
val sut = DefaultElementCallBaseUrlProvider(
|
||||||
|
createFakeElementWellKnownParser(
|
||||||
|
Result.failure(AN_EXCEPTION)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
val matrixClient = FakeMatrixClient(
|
||||||
|
userIdServerNameLambda = userIdServerNameLambda,
|
||||||
|
getUrlLambda = getUrlLambda,
|
||||||
|
)
|
||||||
|
val result = sut.provides(matrixClient)
|
||||||
|
assertThat(result).isNull()
|
||||||
|
userIdServerNameLambda.assertions().isCalledOnce()
|
||||||
|
getUrlLambda.assertions().isCalledOnce()
|
||||||
|
.with(value("https://example.com/.well-known/element/element.json"))
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `provides returns value when getUrl returns correct content`() = runTest {
|
||||||
|
val userIdServerNameLambda = lambdaRecorder<String> { "example.com" }
|
||||||
|
val getUrlLambda = lambdaRecorder<String, Result<String>> { _ ->
|
||||||
|
Result.success("""{"call":{"widget_url":"https://example.com/call"}}""")
|
||||||
|
}
|
||||||
|
val sut = DefaultElementCallBaseUrlProvider(
|
||||||
|
createFakeElementWellKnownParser(
|
||||||
|
Result.success(createElementWellKnown("aUrl"))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
val matrixClient = FakeMatrixClient(
|
||||||
|
userIdServerNameLambda = userIdServerNameLambda,
|
||||||
|
getUrlLambda = getUrlLambda,
|
||||||
|
)
|
||||||
|
val result = sut.provides(matrixClient)
|
||||||
|
assertThat(result).isEqualTo("aUrl")
|
||||||
|
userIdServerNameLambda.assertions().isCalledOnce()
|
||||||
|
getUrlLambda.assertions().isCalledOnce()
|
||||||
|
.with(value("https://example.com/.well-known/element/element.json"))
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun createFakeElementWellKnownParser(result: Result<ElementWellKnown>): FakeElementWellKnownParser {
|
||||||
|
return FakeElementWellKnownParser(result)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun createElementWellKnown(widgetUrl: String): ElementWellKnown {
|
||||||
|
return ElementWellKnown(
|
||||||
|
call = ElementCallWellKnown(
|
||||||
|
widgetUrl = widgetUrl
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -5,7 +5,7 @@
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
* You may obtain a copy of the License at
|
* You may obtain a copy of the License at
|
||||||
*
|
*
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
*
|
*
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
|
@ -14,22 +14,14 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package io.element.android.features.call.impl.wellknown
|
package io.element.android.libraries.matrix.impl.call
|
||||||
|
|
||||||
import kotlinx.serialization.SerialName
|
import org.matrix.rustcomponents.sdk.ElementWellKnown
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
|
|
||||||
/**
|
class FakeElementWellKnownParser(
|
||||||
* Example:
|
private val result: Result<ElementWellKnown>
|
||||||
* <pre>
|
) : ElementWellKnownParser {
|
||||||
* {
|
override fun parse(str: String): Result<ElementWellKnown> {
|
||||||
* "widget_url": "https://call.server.com"
|
return result
|
||||||
* }
|
}
|
||||||
* </pre>
|
}
|
||||||
* .
|
|
||||||
*/
|
|
||||||
@Serializable
|
|
||||||
data class CallWellKnown(
|
|
||||||
@SerialName("widget_url")
|
|
||||||
val widgetUrl: String? = null,
|
|
||||||
)
|
|
||||||
|
|
@ -82,6 +82,8 @@ class FakeMatrixClient(
|
||||||
private val resolveRoomAliasResult: (RoomAlias) -> Result<ResolvedRoomAlias> = { Result.success(ResolvedRoomAlias(A_ROOM_ID, emptyList())) },
|
private val resolveRoomAliasResult: (RoomAlias) -> Result<ResolvedRoomAlias> = { Result.success(ResolvedRoomAlias(A_ROOM_ID, emptyList())) },
|
||||||
private val getRoomPreviewFromRoomIdResult: (RoomId, List<String>) -> Result<RoomPreview> = { _, _ -> Result.failure(AN_EXCEPTION) },
|
private val getRoomPreviewFromRoomIdResult: (RoomId, List<String>) -> Result<RoomPreview> = { _, _ -> Result.failure(AN_EXCEPTION) },
|
||||||
private val clearCacheLambda: () -> Unit = { lambdaError() },
|
private val clearCacheLambda: () -> Unit = { lambdaError() },
|
||||||
|
private val userIdServerNameLambda: () -> String = { lambdaError() },
|
||||||
|
private val getUrlLambda: (String) -> Result<String> = { lambdaError() },
|
||||||
) : MatrixClient {
|
) : MatrixClient {
|
||||||
var setDisplayNameCalled: Boolean = false
|
var setDisplayNameCalled: Boolean = false
|
||||||
private set
|
private set
|
||||||
|
|
@ -315,6 +317,10 @@ class FakeMatrixClient(
|
||||||
override fun sendQueueDisabledFlow(): Flow<RoomId> = sendQueueDisabledFlow
|
override fun sendQueueDisabledFlow(): Flow<RoomId> = sendQueueDisabledFlow
|
||||||
|
|
||||||
override fun userIdServerName(): String {
|
override fun userIdServerName(): String {
|
||||||
TODO("Not yet implemented")
|
return userIdServerNameLambda()
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun getUrl(url: String): Result<String> {
|
||||||
|
return getUrlLambda(url)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue