19
19
import os
20
20
import jwt
21
21
from typing import Optional , List , Literal
22
+ from google .protobuf .json_format import MessageToDict , ParseDict
23
+
24
+ from livekit .protocol .room import RoomConfiguration
22
25
23
26
DEFAULT_TTL = datetime .timedelta (hours = 6 )
24
27
DEFAULT_LEEWAY = datetime .timedelta (minutes = 1 )
27
30
@dataclasses .dataclass
28
31
class VideoGrants :
29
32
# actions on rooms
30
- room_create : bool = False
31
- room_list : bool = False
32
- room_record : bool = False
33
+ room_create : Optional [ bool ] = None
34
+ room_list : Optional [ bool ] = None
35
+ room_record : Optional [ bool ] = None
33
36
34
37
# actions on a particular room
35
- room_admin : bool = False
36
- room_join : bool = False
38
+ room_admin : Optional [ bool ] = None
39
+ room_join : Optional [ bool ] = None
37
40
room : str = ""
38
41
39
42
# permissions within a room
@@ -44,23 +47,22 @@ class VideoGrants:
44
47
# TrackSource types that a participant may publish.
45
48
# When set, it supersedes CanPublish. Only sources explicitly set here can be
46
49
# published
47
- can_publish_sources : List [str ] = dataclasses . field ( default_factory = list )
50
+ can_publish_sources : Optional [ List [str ]] = None
48
51
49
52
# by default, a participant is not allowed to update its own metadata
50
- can_update_own_metadata : bool = False
53
+ can_update_own_metadata : Optional [ bool ] = None
51
54
52
55
# actions on ingresses
53
- ingress_admin : bool = False # applies to all ingress
56
+ ingress_admin : Optional [ bool ] = None # applies to all ingress
54
57
55
58
# participant is not visible to other participants (useful when making bots)
56
- hidden : bool = False
59
+ hidden : Optional [ bool ] = None
57
60
58
- # indicates to the room that current participant is a recorder
59
- recorder : bool = False
61
+ # [deprecated] indicates to the room that current participant is a recorder
62
+ recorder : Optional [ bool ] = None
60
63
61
64
# indicates that the holder can register as an Agent framework worker
62
- # it is also set on all participants that are joining as Agent
63
- agent : bool = False
65
+ agent : Optional [bool ] = None
64
66
65
67
66
68
@dataclasses .dataclass
@@ -75,12 +77,28 @@ class SIPGrants:
75
77
class Claims :
76
78
identity : str = ""
77
79
name : str = ""
78
- video : VideoGrants = dataclasses .field (default_factory = VideoGrants )
79
- sip : SIPGrants = dataclasses .field (default_factory = SIPGrants )
80
- attributes : dict [str , str ] = dataclasses .field (default_factory = dict )
81
- metadata : str = ""
82
- sha256 : str = ""
83
80
kind : str = ""
81
+ metadata : str = ""
82
+ video : Optional [VideoGrants ] = None
83
+ sip : Optional [SIPGrants ] = None
84
+ attributes : Optional [dict [str , str ]] = None
85
+ sha256 : Optional [str ] = None
86
+ room_preset : Optional [str ] = None
87
+ room_config : Optional [RoomConfiguration ] = None
88
+
89
+ def asdict (self ) -> dict :
90
+ # in order to produce minimal JWT size, exclude None or empty values
91
+ claims = dataclasses .asdict (
92
+ self ,
93
+ dict_factory = lambda items : {
94
+ snake_to_lower_camel (k ): v
95
+ for k , v in items
96
+ if v is not None and v != ""
97
+ },
98
+ )
99
+ if self .room_config :
100
+ claims ["roomConfig" ] = MessageToDict (self .room_config )
101
+ return claims
84
102
85
103
86
104
class AccessToken :
@@ -141,16 +159,22 @@ def with_sha256(self, sha256: str) -> "AccessToken":
141
159
self .claims .sha256 = sha256
142
160
return self
143
161
162
+ def with_room_preset (self , preset : str ) -> "AccessToken" :
163
+ self .claims .room_preset = preset
164
+ return self
165
+
166
+ def with_room_config (self , config : RoomConfiguration ) -> "AccessToken" :
167
+ self .claims .room_config = config
168
+ return self
169
+
144
170
def to_jwt (self ) -> str :
145
171
video = self .claims .video
146
- if video .room_join and (not self .identity or not video .room ):
172
+ if video and video .room_join and (not self .identity or not video .room ):
147
173
raise ValueError ("identity and room must be set when joining a room" )
148
174
149
- claims = dataclasses .asdict (
150
- self .claims ,
151
- dict_factory = lambda items : {snake_to_lower_camel (k ): v for k , v in items },
152
- )
153
- claims .update (
175
+ # we want to exclude None values from the token
176
+ jwt_claims = self .claims .asdict ()
177
+ jwt_claims .update (
154
178
{
155
179
"sub" : self .identity ,
156
180
"iss" : self .api_key ,
@@ -164,7 +188,7 @@ def to_jwt(self) -> str:
164
188
),
165
189
}
166
190
)
167
- return jwt .encode (claims , self .api_secret , algorithm = "HS256" )
191
+ return jwt .encode (jwt_claims , self .api_secret , algorithm = "HS256" )
168
192
169
193
170
194
class TokenVerifier :
@@ -208,7 +232,7 @@ def verify(self, token: str) -> Claims:
208
232
}
209
233
sip = SIPGrants (** sip_dict )
210
234
211
- return Claims (
235
+ grant_claims = Claims (
212
236
identity = claims .get ("sub" , "" ),
213
237
name = claims .get ("name" , "" ),
214
238
video = video ,
@@ -218,6 +242,17 @@ def verify(self, token: str) -> Claims:
218
242
sha256 = claims .get ("sha256" , "" ),
219
243
)
220
244
245
+ if claims .get ("roomPreset" ):
246
+ grant_claims .room_preset = claims .get ("roomPreset" )
247
+ if claims .get ("roomConfig" ):
248
+ grant_claims .room_config = ParseDict (
249
+ claims .get ("roomConfig" ),
250
+ RoomConfiguration (),
251
+ ignore_unknown_fields = True ,
252
+ )
253
+
254
+ return grant_claims
255
+
221
256
222
257
def camel_to_snake (t : str ):
223
258
return re .sub (r"(?<!^)(?=[A-Z])" , "_" , t ).lower ()
0 commit comments