# Tokens & Authentication

## Authentication

Stream uses JWT (JSON Web Tokens) to authenticate users so they can open WebSocket connections and send API requests. When a user opens your app, they first pass through your own authentication system. After that, the Stream SDK is initialized and a client instance is created. The device then requests a Stream token from your server. Your server verifies the request and returns a valid token. Once the device receives this token, the user is authenticated and ready to start using chat.

## Generating Tokens

You can generate tokens on the server by creating a Server Client and then using the Create Token method.

If generating a token to use client-side, the token must include the userID claim in the token payload, whereas server tokens do not. When using the create token method, pass the user_id parameter to generate a client-side token.

<codetabs>

<codetabs-item value="python" label="Python">

```python
# pip install getstream
from getstream import Stream

server_client = Stream(
  api_key="{{ api_key }}", api_secret="{{ api_secret }}"
)
token = server_client.create_token("john")
```

</codetabs-item>

<codetabs-item value="ruby" label="Ruby">

```ruby
# gem install getstream-ruby
require 'getstream_ruby'
require 'jwt'

client = GetStreamRuby.manual(api_key: 'STREAM_KEY', api_secret: 'STREAM_SECRET')
token = JWT.encode({ user_id: 'john' }, 'STREAM_SECRET', 'HS256')
```

</codetabs-item>

<codetabs-item value="php" label="PHP">

```php
// composer require getstream/getstream-php

use GetStream\ChatClient;

$client = new ChatClient("{{ api_key }}", "{{ api_secret }}");
$token = $client->createUserToken("john");
```

</codetabs-item>

<codetabs-item value="go" label="Go">

```go
// github.com/GetStream/getstream-go/v4

serverClient, _ := getstream.NewClient(APIKey, APISecret)
token, _ := serverClient.CreateToken("john")
```

</codetabs-item>

<codetabs-item value="csharp" label="C#">

```csharp
// dotnet add package getstream-net
using GetStream;

// Instantiate your Stream client using the API key and secret
// the secret is only used server side and gives you full access to the API.
var client = new StreamClient("{{ api_key }}", "{{ api_secret }}");

var token = client.CreateUserToken("john");
```

</codetabs-item>

<codetabs-item value="nodejs" label="Node">

```js
// Define values.
const api_key = "{{ api_key }}";
const api_secret = "{{ api_secret }}";
const user_id = "john";

// Initialize a Server Client
const serverClient = StreamChat.getInstance(api_key, api_secret);
// Create User Token
const token = serverClient.createToken(user_id);
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
// For Gradle:
// dependencies {
//   implementation "io.getstream:stream-sdk-java:$stream_version"
// }

StreamSDKClient client = new StreamSDKClient("{{ api_key }}", "{{ api_secret }}");

String token = client.tokenBuilder().createToken("john");
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
// User Tokens must be generated server-side, check server-side SDKs for examples
// For Testing/Development - read below how you can enable "Developer Tokens" and skip token generation for development purposes
```

</codetabs-item>

</codetabs>

### Manually Generating Tokens

You can use the JWT generator on this page to generate a User Token JWT without needing to set up a server client. You can use this token for prototyping and debugging; usually by hardcoding this into your application or passing it as an environment value at initialization.

You will need the following values to generate a token:

- `User ID` : A unique string to identify a user.

- `API Secret` : You can find this value in the [Dashboard](https://getstream.io/dashboard/).

To generate a token, provide a `User ID` and your `API Secret` to the following generator:

<token-generator></token-generator>

For more information on how JWT works, please visit <https://jwt.io>.

## Setting Automatic Token Expiration

By default, user tokens are valid indefinitely. You can set an expiration to tokens by passing it as the second parameter. The expiration should contain the number of seconds since Unix epoch (00:00:00 UTC on 1 January 1970).

<codetabs>

<codetabs-item value="javascript" label="JavaScript">

```js
// creates a token that expires in 1 hour using moment.js
const timestamp = Number(moment().add("1h").format("X"));
const token1 = client.createToken("john", timestamp);

// the same can be done with plain javascript
const token2 = client.createToken(
  "john",
  Math.floor(Date.now() / 1000) + 60 * 60,
);
```

</codetabs-item>

<codetabs-item value="python" label="Python">

```python
# creates a token valid for 1 hour
token = server_client.create_token("john", expiration=3600)
```

</codetabs-item>

<codetabs-item value="ruby" label="Ruby">

```ruby
# creates a token valid for 1 hour
require 'jwt'

token = JWT.encode(
  { user_id: 'john', exp: Time.now.to_i + 3600 },
  api_secret,
  'HS256'
)
```

</codetabs-item>

<codetabs-item value="php" label="PHP">

```php
// creates a token valid for 1 hour
$expiration = (new \DateTime())->getTimestamp() + 3600;
$token = $client->createUserToken("john", expiration: $expiration);
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
// creates a token valid for 1 hour
String token = client.tokenBuilder().createToken("john", 3600);
```

</codetabs-item>

<codetabs-item value="go" label="Go">

```go
// creates a token valid for 1 hour
token, _ := client.CreateToken("john", getstream.WithExpiration(time.Hour))
```

</codetabs-item>

<codetabs-item value="swift" label="Swift">

```swift
// at the moment we don't have a Swift client for server side usage
// You can use our user token generator here during development: https://getstream.io/chat/docs/token_generator/?language=swift
```

</codetabs-item>

<codetabs-item value="csharp" label="C#">

```csharp
// creates a token valid for 1 hour
var token = client.CreateUserToken("john", expiration: TimeSpan.FromHours(1));
```

</codetabs-item>

<codetabs-item value="dart" label="Dart">

```dart
// user tokens must be generated server-side, check other languages for examples
```

</codetabs-item>

<codetabs-item value="unreal" label="Unreal">

```cpp
// user tokens must be generated server-side, check other languages for examples
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
// User Tokens must be generated server-side, check server-side SDKs for examples
// For Testing/Development - read below how you can enable "Developer Tokens" and skip token generation for development purposes
```

</codetabs-item>

</codetabs>

## Token Providers

A concept we will refer to throughout the docs is a Token Provider. At a high level, the Token Provider is an endpoint on your server that can perform the following sequence of tasks:

1. Receive information about a user from the front end.

2. Validate that user information against your system.

3. Provide a User-ID corresponding to that user to the server client's token creation method.

4. Return that token to the front end.

To learn more about Token Providers, read on in our [Initialization & Users](/chat/docs/<framework>/init_and_users/) section.

## Developer Tokens

For development applications, it is possible to disable token authentication and use client-side generated tokens or a manually generated static token. Disabling auth checks is not suitable for a production application and should only be done for proofs-of-concept and applications in the early development stage. To enable development tokens, you need to change your application configuration.

On the [Dashboard](https://getstream.io/dashboard/):

1. Select the App you want to enable developer tokens on and ensure it is in Development mode

2. Click _App_ name to enter the _Chat Overview_

3. Scroll to the _General_ section

4. Toggle _Disable Authentication Checks_

This disables the authentication check, but does not remove the requirement to send a token. Send either a client generated development token, or manually create one and hard code it into your application.

<codetabs>

<codetabs-item value="kotlin" label="Kotlin">

```kotlin
val user = User(
  id = "bender",
  name = "Bender",
  image = "https://bit.ly/321RmWb",
)
val token = client.devToken(user.id)

client.connectUser(user, token).enqueue { /* ... */ }
```

</codetabs-item>

<codetabs-item value="javascript" label="JavaScript">

```js
await client.connectUser(
  {
    id: "john",
    name: "John Doe",
    image: "https://getstream.io/random_svg/?name=John",
  },
  client.devToken("john"),
);
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
User user = new User();
user.setId("bender");
user.setName("Bender");
user.setImage("https://bit.ly/321RmWb");

String token = client.devToken(user.getId());

client.connectUser(user, token).enqueue(result -> { /* ... */ });
```

</codetabs-item>

<codetabs-item value="swift" label="Swift">

```swift
import StreamChat

client.connectUser(userInfo: .init(id: "john-doe"), token: .development(userId: "john-doe"))
```

</codetabs-item>

<codetabs-item value="dart" label="Dart">

```dart
final user = User(id: "john", extraData: {
 "name": "John Doe",
 "image": "https://getstream.io/random_svg/?name=John",
});

await client.setUser(
 user,
 client.devToken("john"),
);
```

</codetabs-item>

<codetabs-item value="unreal" label="Unreal">

```cpp
const FUser User{TEXT("john")};
const FString Token = Client->DevToken(User.Id);
Client->ConnectUser(
  User,
  Token,
  [](const FOwnUser& UserRef)
  {
    // Connection established
  });
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
var userName = "The Amazing Tom";
var userId = StreamChatClient.SanitizeUserId(userName); // Remove disallowed characters
var userToken = StreamChatClient.CreateDeveloperAuthToken(userId);
var credentials = new AuthCredentials("API_KEY", userId, userToken);

// Create chat client
var client = StreamChatClient.CreateDefaultClient();

// Connect user
var localUserData = await client.ConnectUserAsync(credentials);
```

</codetabs-item>

</codetabs>

## Manual Token Expiration

Token Revocation is a way to manually expire tokens for a single user or for many users by setting a `revoke_tokens_issued_before` time, and any tokens issued before this will be considered expired and will fail to authenticate.  This can be reversed by setting the field to null.

### Token Revocation by User

You can revoke all tokens that belong to a certain user or a list of users.

<codetabs>

<codetabs-item value="javascript" label="JavaScript">

```js
await client.revokeUserToken("user-id", revokeDate);
await client.revokeUsersToken(["user1-id", "user2-id"], revokeDate);
```

</codetabs-item>

<codetabs-item value="python" label="Python">

```python
from getstream.models import UpdateUserPartialRequest

client.update_users_partial(users=[
    UpdateUserPartialRequest(id="user-id", set={"revoke_tokens_issued_before": revoke_date})
])
client.update_users_partial(users=[
    UpdateUserPartialRequest(id=uid, set={"revoke_tokens_issued_before": revoke_date})
    for uid in ["user1-id", "user2-id"]
])
```

</codetabs-item>

<codetabs-item value="go" label="Go">

```go
client.UpdateUsersPartial(ctx, &getstream.UpdateUsersPartialRequest{
	Users: []getstream.UpdateUserPartialRequest{
		{ID: "user-id", Set: map[string]any{"revoke_tokens_issued_before": revokeTime}},
	},
})
client.UpdateUsersPartial(ctx, &getstream.UpdateUsersPartialRequest{
	Users: []getstream.UpdateUserPartialRequest{
		{ID: "user1-id", Set: map[string]any{"revoke_tokens_issued_before": revokeTime}},
		{ID: "user2-id", Set: map[string]any{"revoke_tokens_issued_before": revokeTime}},
	},
})
```

</codetabs-item>

<codetabs-item value="php" label="PHP">

```php
use GetStream\GeneratedModels as Models;

$client->updateUsersPartial(new Models\UpdateUsersPartialRequest(
    users: [new Models\UpdateUserPartialRequest(
        id: "user-id",
        set: (object)["revoke_tokens_issued_before" => (new \DateTime())->format(\DateTime::ATOM)],
    )]
));

$client->updateUsersPartial(new Models\UpdateUsersPartialRequest(
    users: [
        new Models\UpdateUserPartialRequest(
            id: "user1-id",
            set: (object)["revoke_tokens_issued_before" => (new \DateTime())->format(\DateTime::ATOM)],
        ),
        new Models\UpdateUserPartialRequest(
            id: "user2-id",
            set: (object)["revoke_tokens_issued_before" => (new \DateTime())->format(\DateTime::ATOM)],
        ),
    ]
));
```

</codetabs-item>

<codetabs-item value="ruby" label="Ruby">

```ruby
Models = GetStream::Generated::Models

client.common.update_users_partial(
  Models::UpdateUsersPartialRequest.new(
    users: [Models::UpdateUserPartialRequest.new(
      id: 'user-id', set: { 'revoke_tokens_issued_before' => before.iso8601 }
    )]
  )
)
client.common.update_users_partial(
  Models::UpdateUsersPartialRequest.new(
    users: ['user1-id', 'user2-id'].map { |uid|
      Models::UpdateUserPartialRequest.new(
        id: uid, set: { 'revoke_tokens_issued_before' => before.iso8601 }
      )
    }
  )
)
# before should be a Time object
```

</codetabs-item>

<codetabs-item value="csharp" label="C#">

```csharp
// dotnet add package getstream-net
using GetStream;
using GetStream.Models;

var client = new StreamClient("{{ api_key }}", "{{ api_secret }}");

await client.UpdateUsersPartialAsync(new UpdateUsersPartialRequest
{
    Users = new List<UpdateUserPartialRequest>
    {
        new UpdateUserPartialRequest
        {
            ID = "<user-id>",
            Set = new Dictionary<string, object>
            {
                ["revoke_tokens_issued_before"] = DateTimeOffset.UtcNow.AddHours(-1)
            }
        }
    }
});

// Revoke tokens for multiple users
await client.UpdateUsersPartialAsync(new UpdateUsersPartialRequest
{
    Users = new List<UpdateUserPartialRequest>
    {
        new UpdateUserPartialRequest
        {
            ID = "user1-id",
            Set = new Dictionary<string, object>
            {
                ["revoke_tokens_issued_before"] = DateTimeOffset.UtcNow.AddHours(-1)
            }
        },
        new UpdateUserPartialRequest
        {
            ID = "user2-id",
            Set = new Dictionary<string, object>
            {
                ["revoke_tokens_issued_before"] = DateTimeOffset.UtcNow.AddHours(-1)
            }
        }
    }
});
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
client.updateUsersPartial(UpdateUsersPartialRequest.builder()
    .users(List.of(UpdateUserPartialRequest.builder()
        .id("<user-id>")
        .set(Map.of("revoke_tokens_issued_before", new Date()))
        .build()))
    .build()).execute();

client.updateUsersPartial(UpdateUsersPartialRequest.builder()
    .users(List.of(
        UpdateUserPartialRequest.builder()
            .id("<user1-id>")
            .set(Map.of("revoke_tokens_issued_before", new Date()))
            .build(),
        UpdateUserPartialRequest.builder()
            .id("<user2-id>")
            .set(Map.of("revoke_tokens_issued_before", new Date()))
            .build()))
    .build()).execute();
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
// This is a server-side feature
```

</codetabs-item>

</codetabs>

Note: Your tokens must include the `iat` (issued at time) claim, which will be compared to the time in the `revoke_tokens_issued_before` field to determine whether the token is valid or expired.  Tokens which have no `iat` will be considered invalid.

### Undoing the revoke

To undo user-level token revocation, you can simply set revocation date to `null`:

<codetabs>

<codetabs-item value="javascript" label="JavaScript">

```js
await client.revokeUserToken("user-id", null);
await client.revokeUsersToken(["user1-id", "user2-id"], null);
```

</codetabs-item>

<codetabs-item value="python" label="Python">

```python
from getstream.models import UpdateUserPartialRequest

client.update_users_partial(users=[
    UpdateUserPartialRequest(id="user-id", unset=["revoke_tokens_issued_before"])
])
client.update_users_partial(users=[
    UpdateUserPartialRequest(id=uid, unset=["revoke_tokens_issued_before"])
    for uid in ["user1-id", "user2-id"]
])
```

</codetabs-item>

<codetabs-item value="go" label="Go">

```go
client.UpdateUsersPartial(ctx, &getstream.UpdateUsersPartialRequest{
	Users: []getstream.UpdateUserPartialRequest{
		{ID: "user-id", Unset: []string{"revoke_tokens_issued_before"}},
	},
})
client.UpdateUsersPartial(ctx, &getstream.UpdateUsersPartialRequest{
	Users: []getstream.UpdateUserPartialRequest{
		{ID: "user1-id", Unset: []string{"revoke_tokens_issued_before"}},
		{ID: "user2-id", Unset: []string{"revoke_tokens_issued_before"}},
	},
})
```

</codetabs-item>

<codetabs-item value="php" label="PHP">

```php
use GetStream\GeneratedModels as Models;

$client->updateUsersPartial(new Models\UpdateUsersPartialRequest(
    users: [new Models\UpdateUserPartialRequest(
        id: "user-id",
        unset: ["revoke_tokens_issued_before"],
    )]
));

$client->updateUsersPartial(new Models\UpdateUsersPartialRequest(
    users: [
        new Models\UpdateUserPartialRequest(
            id: "user1-id",
            unset: ["revoke_tokens_issued_before"],
        ),
        new Models\UpdateUserPartialRequest(
            id: "user2-id",
            unset: ["revoke_tokens_issued_before"],
        ),
    ]
));
```

</codetabs-item>

<codetabs-item value="ruby" label="Ruby">

```ruby
Models = GetStream::Generated::Models

client.common.update_users_partial(
  Models::UpdateUsersPartialRequest.new(
    users: [Models::UpdateUserPartialRequest.new(
      id: 'user-id', unset: ['revoke_tokens_issued_before']
    )]
  )
)
client.common.update_users_partial(
  Models::UpdateUsersPartialRequest.new(
    users: ['user1-id', 'user2-id'].map { |uid|
      Models::UpdateUserPartialRequest.new(
        id: uid, unset: ['revoke_tokens_issued_before']
      )
    }
  )
)
```

</codetabs-item>

<codetabs-item value="csharp" label="C#">

```csharp
// dotnet add package getstream-net
using GetStream;
using GetStream.Models;

var client = new StreamClient("{{ api_key }}", "{{ api_secret }}");

await client.UpdateUsersPartialAsync(new UpdateUsersPartialRequest
{
    Users = new List<UpdateUserPartialRequest>
    {
        new UpdateUserPartialRequest
        {
            ID = "<user-id>",
            Unset = new List<string> { "revoke_tokens_issued_before" }
        }
    }
});

// Undo revoke for multiple users
await client.UpdateUsersPartialAsync(new UpdateUsersPartialRequest
{
    Users = new List<UpdateUserPartialRequest>
    {
        new UpdateUserPartialRequest
        {
            ID = "user1-id",
            Unset = new List<string> { "revoke_tokens_issued_before" }
        },
        new UpdateUserPartialRequest
        {
            ID = "user2-id",
            Unset = new List<string> { "revoke_tokens_issued_before" }
        }
    }
});
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
client.updateUsersPartial(UpdateUsersPartialRequest.builder()
    .users(List.of(UpdateUserPartialRequest.builder()
        .id("<user-id>")
        .unset(List.of("revoke_tokens_issued_before"))
        .build()))
    .build()).execute();

client.updateUsersPartial(UpdateUsersPartialRequest.builder()
    .users(List.of(
        UpdateUserPartialRequest.builder()
            .id("<user1-id>")
            .unset(List.of("revoke_tokens_issued_before"))
            .build(),
        UpdateUserPartialRequest.builder()
            .id("<user2-id>")
            .unset(List.of("revoke_tokens_issued_before"))
            .build()))
    .build()).execute();
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
// This is a server-side feature
```

</codetabs-item>

</codetabs>

### Token Revocation by Application

It is possible to revoke tokens for all users of an application. This should be used with caution as it will expire every user's token, regardless of whether the token has an `iat` claim.

<codetabs>

<codetabs-item value="javascript" label="JavaScript">

```js
await client.revokeTokens(revokeDate);
// you can pass Date or ISOstring as value here
```

</codetabs-item>

<codetabs-item value="python" label="Python">

```python
client.update_app(revoke_tokens_issued_before=revoke_time)
# revoke_time is a datetime object
```

</codetabs-item>

<codetabs-item value="go" label="Go">

```go
client.UpdateApp(ctx, &getstream.UpdateAppRequest{
	RevokeTokensIssuedBefore: &revokeTime,
})
// revokeTime is a time.Time object
```

</codetabs-item>

<codetabs-item value="php" label="PHP">

```php
use GetStream\GeneratedModels as Models;

$client->updateApp(new Models\UpdateAppRequest(
    revokeTokensIssuedBefore: new \DateTime("now"),
));
// revokeTokensIssuedBefore is a DateTime object
```

</codetabs-item>

<codetabs-item value="ruby" label="Ruby">

```ruby
Models = GetStream::Generated::Models

client.common.update_app(
  Models::UpdateAppRequest.new(revoke_tokens_issued_before: before)
)
# before is a Time object
```

</codetabs-item>

<codetabs-item value="csharp" label="C#">

```csharp
// dotnet add package getstream-net
using GetStream;
using GetStream.Models;

var client = new StreamClient("{{ api_key }}", "{{ api_secret }}");

await client.UpdateAppAsync(new UpdateAppRequest
{
    RevokeTokensIssuedBefore = DateTimeOffset.UtcNow.AddHours(-1).UtcDateTime
});
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
// Revocation date must be at least 60 seconds in the past
Calendar cal = Calendar.getInstance();
cal.add(Calendar.MINUTE, -2);
client.updateApp(UpdateAppRequest.builder()
    .revokeTokensIssuedBefore(cal.getTime())
    .build()).execute();
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
// This is a server-side feature
```

</codetabs-item>

</codetabs>

### Undoing the revoke

To undo app-level token revocation, you can simply set revocation date to `null`:

<codetabs>

<codetabs-item value="javascript" label="JavaScript">

```js
await client.revokeTokens(null);
```

</codetabs-item>

<codetabs-item value="python" label="Python">

```python
client.update_app(revoke_tokens_issued_before=None)
```

</codetabs-item>

<codetabs-item value="go" label="Go">

```go
client.UpdateApp(ctx, &getstream.UpdateAppRequest{
	RevokeTokensIssuedBefore: nil,
})
```

</codetabs-item>

<codetabs-item value="php" label="PHP">

```php
use GetStream\GeneratedModels as Models;

$client->updateApp(new Models\UpdateAppRequest(
    revokeTokensIssuedBefore: null,
));
```

</codetabs-item>

<codetabs-item value="ruby" label="Ruby">

```ruby
Models = GetStream::Generated::Models

client.common.update_app(
  Models::UpdateAppRequest.new(revoke_tokens_issued_before: nil)
)
```

</codetabs-item>

<codetabs-item value="csharp" label="C#">

```csharp
// dotnet add package getstream-net
using GetStream;
using GetStream.Models;

var client = new StreamClient("{{ api_key }}", "{{ api_secret }}");

await client.UpdateAppAsync(new UpdateAppRequest
{
    RevokeTokensIssuedBefore = null
});
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
client.updateApp(UpdateAppRequest.builder()
    .revokeTokensIssuedBefore(null)
    .build()).execute();
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
// This is a server-side feature
```

</codetabs-item>

</codetabs>

### Adding iat claim to token

By default, user tokens generated through the createToken function do not contain information about time of issue. You can change that by passing the issue date as the third parameter while creating tokens. This is a security best practice, as it enables revoking tokens.

<codetabs>

<codetabs-item value="javascript" label="JavaScript">

```js
client.createToken("user-id", expireTime, issuedAt);
// issuedAt should be unix timestamp
// issuedAt = Math.floor(Date.now() / 1000)
```

</codetabs-item>

<codetabs-item value="python" label="Python">

```python
token = server_client.create_token("user-id", expiration=expiry_seconds)
# The token automatically includes the iat (issued at) claim
```

</codetabs-item>

<codetabs-item value="go" label="Go">

```go
client.CreateToken("user-id",
	getstream.WithExpiration(expiryTime),
)
// expiryTime is a time.Duration (e.g. 24 * time.Hour)
```

</codetabs-item>

<codetabs-item value="php" label="PHP">

```php
$token = $client->createUserToken("user-id", expiration: $expiry, claims: ["iat" => $issuedAt]);
// issuedAt should be unix timestamp
```

</codetabs-item>

<codetabs-item value="ruby" label="Ruby">

```ruby
require 'jwt'

token = JWT.encode(
  { user_id: 'john', exp: exp_time, iat: issued_at },
  api_secret,
  'HS256'
)
# issued_at should be a unix timestamp
```

</codetabs-item>

<codetabs-item value="csharp" label="C#">

```csharp
// creates a token valid for 2 hours
// The token automatically includes the iat (issued at) claim
var token = client.CreateUserToken("user1-id", expiration: TimeSpan.FromHours(2));
```

</codetabs-item>

<codetabs-item value="java" label="Java">

```java
// creates a token valid for 1 hour
// The token automatically includes the iat (issued at) claim
String token = client.tokenBuilder().createToken("john", 3600);
```

</codetabs-item>

<codetabs-item value="unity" label="Unity">

```csharp
// This is a server-side feature
```

</codetabs-item>

</codetabs>


---

This page was last updated at 2026-03-13T13:17:13.197Z.

For the most recent version of this documentation, visit [https://getstream.io/chat/docs/unity/tokens_and_authentication/](https://getstream.io/chat/docs/unity/tokens_and_authentication/).