Skip to content

Commit 5f277f9

Browse files
authored
Add ability to set room configuration on access token (#105)
1 parent 6f47ecc commit 5f277f9

File tree

11 files changed

+150
-36
lines changed

11 files changed

+150
-36
lines changed

.changeset/poor-dolphins-explain.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
"server-sdk-kotlin": patch
3+
---
4+
5+
Add ability to set room configuration on access token

.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ local.properties
1313
.idea/kotlinScripting.xml
1414
.idea/libraries/
1515
.idea/migrations.xml
16+
.idea/protoeditor.xml
17+
.idea/runConfigurations.xml
18+
.idea/caches/
1619
.idea/**/gradle.xml
1720
*.iws
1821
*.iml

.idea/misc.xml

+2-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

build.gradle.kts

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ repositories {
1919
mavenCentral()
2020
}
2121
apply(from = "gradle/gradle-mvn-push.gradle")
22+
apply(plugin = "idea")
2223

2324
plugins {
2425
kotlin("jvm") version "1.9.0"
@@ -30,7 +31,6 @@ plugins {
3031
id("com.diffplug.spotless") version "6.21.0"
3132
}
3233

33-
3434
java {
3535
withJavadocJar()
3636
withSourcesJar()

gradle/wrapper/gradle-wrapper.jar

-16.8 KB
Binary file not shown.
+4-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
distributionBase=GRADLE_USER_HOME
22
distributionPath=wrapper/dists
3-
distributionUrl=https\://services.gradle.org/distributions/gradle-7.4.2-bin.zip
3+
distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip
4+
networkTimeout=10000
5+
validateDistributionUrl=true
46
zipStoreBase=GRADLE_USER_HOME
5-
zipStorePath=wrapper/dists
7+
zipStorePath=wrapper/dists

gradlew

+31-13
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@
1515
# See the License for the specific language governing permissions and
1616
# limitations under the License.
1717
#
18+
# SPDX-License-Identifier: Apache-2.0
19+
#
1820

1921
##############################################################################
2022
#
@@ -55,7 +57,7 @@
5557
# Darwin, MinGW, and NonStop.
5658
#
5759
# (3) This script is generated from the Groovy template
58-
# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
60+
# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
5961
# within the Gradle project.
6062
#
6163
# You can find Gradle at https://github.com/gradle/gradle/.
@@ -80,13 +82,12 @@ do
8082
esac
8183
done
8284

83-
APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit
84-
85-
APP_NAME="Gradle"
85+
# This is normally unused
86+
# shellcheck disable=SC2034
8687
APP_BASE_NAME=${0##*/}
87-
88-
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
89-
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
88+
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
89+
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s
90+
' "$PWD" ) || exit
9091

9192
# Use the maximum available, or set MAX_FD != -1 to use that value.
9293
MAX_FD=maximum
@@ -133,22 +134,29 @@ location of your Java installation."
133134
fi
134135
else
135136
JAVACMD=java
136-
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
137+
if ! command -v java >/dev/null 2>&1
138+
then
139+
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
137140
138141
Please set the JAVA_HOME variable in your environment to match the
139142
location of your Java installation."
143+
fi
140144
fi
141145

142146
# Increase the maximum file descriptors if we can.
143147
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
144148
case $MAX_FD in #(
145149
max*)
150+
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
151+
# shellcheck disable=SC2039,SC3045
146152
MAX_FD=$( ulimit -H -n ) ||
147153
warn "Could not query maximum file descriptor limit"
148154
esac
149155
case $MAX_FD in #(
150156
'' | soft) :;; #(
151157
*)
158+
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
159+
# shellcheck disable=SC2039,SC3045
152160
ulimit -n "$MAX_FD" ||
153161
warn "Could not set maximum file descriptor limit to $MAX_FD"
154162
esac
@@ -193,18 +201,28 @@ if "$cygwin" || "$msys" ; then
193201
done
194202
fi
195203

196-
# Collect all arguments for the java command;
197-
# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of
198-
# shell script including quotes and variable substitutions, so put them in
199-
# double quotes to make sure that they get re-expanded; and
200-
# * put everything else in single quotes, so that it's not re-expanded.
204+
205+
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
206+
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
207+
208+
# Collect all arguments for the java command:
209+
# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
210+
# and any embedded shellness will be escaped.
211+
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
212+
# treated as '${Hostname}' itself on the command line.
201213

202214
set -- \
203215
"-Dorg.gradle.appname=$APP_BASE_NAME" \
204216
-classpath "$CLASSPATH" \
205217
org.gradle.wrapper.GradleWrapperMain \
206218
"$@"
207219

220+
# Stop when "xargs" is not available.
221+
if ! command -v xargs >/dev/null 2>&1
222+
then
223+
die "xargs is not available"
224+
fi
225+
208226
# Use "xargs" to parse quoted args.
209227
#
210228
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.

gradlew.bat

+21-16
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,10 @@
1313
@rem See the License for the specific language governing permissions and
1414
@rem limitations under the License.
1515
@rem
16+
@rem SPDX-License-Identifier: Apache-2.0
17+
@rem
1618

17-
@if "%DEBUG%" == "" @echo off
19+
@if "%DEBUG%"=="" @echo off
1820
@rem ##########################################################################
1921
@rem
2022
@rem Gradle startup script for Windows
@@ -25,7 +27,8 @@
2527
if "%OS%"=="Windows_NT" setlocal
2628

2729
set DIRNAME=%~dp0
28-
if "%DIRNAME%" == "" set DIRNAME=.
30+
if "%DIRNAME%"=="" set DIRNAME=.
31+
@rem This is normally unused
2932
set APP_BASE_NAME=%~n0
3033
set APP_HOME=%DIRNAME%
3134

@@ -40,13 +43,13 @@ if defined JAVA_HOME goto findJavaFromJavaHome
4043

4144
set JAVA_EXE=java.exe
4245
%JAVA_EXE% -version >NUL 2>&1
43-
if "%ERRORLEVEL%" == "0" goto execute
46+
if %ERRORLEVEL% equ 0 goto execute
4447

45-
echo.
46-
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
47-
echo.
48-
echo Please set the JAVA_HOME variable in your environment to match the
49-
echo location of your Java installation.
48+
echo. 1>&2
49+
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
50+
echo. 1>&2
51+
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
52+
echo location of your Java installation. 1>&2
5053

5154
goto fail
5255

@@ -56,11 +59,11 @@ set JAVA_EXE=%JAVA_HOME%/bin/java.exe
5659

5760
if exist "%JAVA_EXE%" goto execute
5861

59-
echo.
60-
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
61-
echo.
62-
echo Please set the JAVA_HOME variable in your environment to match the
63-
echo location of your Java installation.
62+
echo. 1>&2
63+
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
64+
echo. 1>&2
65+
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
66+
echo location of your Java installation. 1>&2
6467

6568
goto fail
6669

@@ -75,13 +78,15 @@ set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
7578

7679
:end
7780
@rem End local scope for the variables with windows NT shell
78-
if "%ERRORLEVEL%"=="0" goto mainEnd
81+
if %ERRORLEVEL% equ 0 goto mainEnd
7982

8083
:fail
8184
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
8285
rem the _cmd.exe /c_ return code!
83-
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
84-
exit /b 1
86+
set EXIT_CODE=%ERRORLEVEL%
87+
if %EXIT_CODE% equ 0 set EXIT_CODE=1
88+
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
89+
exit /b %EXIT_CODE%
8590

8691
:mainEnd
8792
if "%OS%"=="Windows_NT" endlocal

protocol

Submodule protocol updated 70 files

src/main/kotlin/io/livekit/server/AccessToken.kt

+32-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 LiveKit, Inc.
2+
* Copyright 2024-2025 LiveKit, Inc.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -19,6 +19,8 @@ package io.livekit.server
1919
import com.auth0.jwt.JWT
2020
import com.auth0.jwt.JWTCreator
2121
import com.auth0.jwt.algorithms.Algorithm
22+
import com.google.protobuf.MessageOrBuilder
23+
import livekit.LivekitRoom.RoomConfiguration
2224
import java.time.Instant
2325
import java.util.Date
2426
import java.util.concurrent.TimeUnit
@@ -84,6 +86,18 @@ class AccessToken(
8486
*/
8587
val attributes = mutableMapOf<String, String>()
8688

89+
/**
90+
* Use a named preset room configuration.
91+
*
92+
* Any options set in [roomConfiguration] will take precedence.
93+
*/
94+
var roomPreset: String? = null
95+
96+
/**
97+
* Configuration for when creating a room.
98+
*/
99+
var roomConfiguration: RoomConfiguration? = null
100+
87101
/**
88102
* Add [VideoGrant] to this token.
89103
*/
@@ -166,11 +180,14 @@ class AccessToken(
166180
name?.let { claimsMap["name"] = it }
167181
metadata?.let { claimsMap["metadata"] = it }
168182
sha256?.let { claimsMap["sha256"] = it }
183+
roomPreset?.let { claimsMap["roomPreset"] = it }
169184
attributes.toMap().let { attributesCopy ->
170185
if (attributesCopy.isNotEmpty()) {
171186
claimsMap["attributes"] = attributesCopy
172187
}
173188
}
189+
roomConfiguration?.let { claimsMap["roomConfig"] = it.toMap() }
190+
174191
claimsMap["video"] = videoGrantsMap
175192
claimsMap["sip"] = sipGrantsMap
176193

@@ -202,3 +219,17 @@ internal fun JWTCreator.Builder.withClaimAny(name: String, value: Any) {
202219
}
203220
}
204221
}
222+
223+
internal fun MessageOrBuilder.toMap(): Map<String, *> {
224+
val map = mutableMapOf<String, Any>()
225+
226+
for ((field, value) in allFields) {
227+
if (value is MessageOrBuilder) {
228+
map[field.name] = value.toMap()
229+
} else {
230+
map[field.name] = value
231+
}
232+
}
233+
234+
return map
235+
}

src/test/kotlin/io/livekit/server/AccessTokenTest.kt

+50-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2024 LiveKit, Inc.
2+
* Copyright 2024-2025 LiveKit, Inc.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -18,6 +18,7 @@ package io.livekit.server
1818

1919
import com.auth0.jwt.JWT
2020
import com.auth0.jwt.algorithms.Algorithm
21+
import livekit.LivekitRoom.RoomConfiguration
2122
import org.junit.jupiter.api.Test
2223
import java.util.Date
2324
import kotlin.test.assertEquals
@@ -34,12 +35,32 @@ class AccessTokenTest {
3435
fun createToken() {
3536
val token = AccessToken(KEY, SECRET)
3637

38+
val roomConfig = with(RoomConfiguration.newBuilder()) {
39+
name = "name"
40+
emptyTimeout = 1
41+
departureTimeout = 2
42+
maxParticipants = 3
43+
minPlayoutDelay = 4
44+
maxPlayoutDelay = 5
45+
egress = with(egressBuilder) {
46+
room = with(roomBuilder) {
47+
roomName = "name"
48+
buildPartial()
49+
}
50+
buildPartial()
51+
}
52+
syncStreams = true
53+
build()
54+
}
55+
3756
token.expiration = Date(33254282804000) // 10/15/3023
3857
token.name = "name"
3958
token.identity = "identity"
4059
token.metadata = "metadata"
4160
token.sha256 = "gfedcba"
4261
token.attributes["key"] = "value"
62+
token.roomPreset = "roomPreset"
63+
token.roomConfiguration = roomConfig
4364

4465
token.addGrants(RoomName("room_name"))
4566
token.addGrants(CanPublishSources(listOf("camera", "microphone")))
@@ -60,8 +81,10 @@ class AccessTokenTest {
6081
assertEquals(token.identity, claims["jti"]?.asString())
6182
assertEquals(token.metadata, claims["metadata"]?.asString())
6283
assertEquals(token.sha256, claims["sha256"]?.asString())
84+
assertEquals(token.roomPreset, claims["roomPreset"]?.asString())
6385
assertEquals(token.expiration, decodedJWT.expiresAt)
6486
assertEquals(token.attributes["key"], claims["attributes"]?.asMap()?.get("key"))
87+
assertEquals(token.roomConfiguration?.toMap(), claims["roomConfig"]?.asMap())
6588

6689
val videoGrants = claims["video"]?.asMap()
6790
assertNotNull(videoGrants)
@@ -72,4 +95,30 @@ class AccessTokenTest {
7295
assertNotNull(sipGrants)
7396
assertEquals(true, sipGrants["admin"])
7497
}
98+
99+
@Test
100+
fun protobufMapConversion() {
101+
val roomConfig = with(RoomConfiguration.newBuilder()) {
102+
name = "name"
103+
emptyTimeout = 1
104+
departureTimeout = 2
105+
maxParticipants = 3
106+
minPlayoutDelay = 4
107+
maxPlayoutDelay = 5
108+
egress = with(egressBuilder) {
109+
room = with(roomBuilder) {
110+
roomName = "name"
111+
buildPartial()
112+
}
113+
buildPartial()
114+
}
115+
syncStreams = true
116+
build()
117+
}
118+
119+
val map = roomConfig.toMap()
120+
121+
assertEquals(roomConfig.emptyTimeout, map["empty_timeout"])
122+
assertEquals(roomConfig.egress.room.roomName, ((map["egress"] as Map<*, *>)["room"] as Map<*, *>)["room_name"])
123+
}
75124
}

0 commit comments

Comments
 (0)