Audio Player

Audio players can be used to play audio across voice connections. A single audio player can play the same audio over multiple voice connections.

Cheat sheet

Creation

Creating an audio player is simple:

const { createAudioPlayer } = require('@discordjs/voice');

const player = createAudioPlayer();
1
2
3

You can also customize the behaviors of an audio player. For example, the default behavior is to pause when there are no active subscribers for an audio player. This behavior can be configured to either pause, stop, or just continue playing through the stream:

const { createAudioPlayer, NoSubscriberBehavior } = require('@discordjs/voice');

const player = createAudioPlayer({
	behaviors: {
		noSubscriber: NoSubscriberBehavior.Pause,
	},
});
1
2
3
4
5
6
7

Deletion

If you no longer require an audio player, you can stop() it and then remove references to it so that it gets garbage collected.

player.stop();
1

Playing audio

You can create audio resources and then play them on an audio player.

const resource = createAudioResource('/home/user/voice/track.mp3');
player.play(resource);

// Play "track.mp3" across two voice connections
connection1.subscribe(player);
connection2.subscribe(player);
1
2
3
4
5
6

WARNING

Audio players can play one audio resource at most. If you try to play another audio resource while one is already playing on the same player, the existing one is destroyed and replaced with the new one.

Pausing/unpausing

You can call the pause() and unpause() methods. While the audio player is paused, no audio will be played. When it is resumed, it will continue where it left off.

player.pause();

// Unpause after 5 seconds
setTimeout(() => player.unpause(), 5_000);
1
2
3
4

Life cycle

Voice connections have their own life cycle, with five distinct states. You can follow the methods discussed in the life cycles section to subscribe to changes to voice connections.

  • Idle - the initial state of an audio player. The audio player will be in this state when there is no audio resource for it to play.

  • Bufferring - the state an audio player will be in while it is waiting for an audio resource to become playable. The audio player may transition from this state to either the Playing state (success) or the Idle state (failure).

  • Playing - the state a voice connection enters when it is actively playing an audio resource. When the audio resource comes to an end, the audio player will transition to the Idle state.

  • AutoPaused - the state a voice connection will enter when the player has paused itself because there are no active voice connections to play to. This is only possible with the noSubscriber behavior set to Pause. It will automatically transition back to Playing once at least one connection becomes available again.

  • Paused - the state a voice connection enters when it is paused by the user.

const { AudioPlayerStatus } = require('@discordjs/voice');

player.on(AudioPlayerStatus.Playing, () => {
	console.log('The audio player has started playing!');
});
1
2
3
4
5

Handling errors

When an audio player is given an audio resource to play, it will propagate errors from the audio resource for you to handle.

In the error handler, you can choose to either play a new audio resource or stop playback. If you take no action, the audio player will stop itself and revert to the Idle state.

Additionally, the error object will also contain a resource property that helps you to figure out which audio resource created the error.

Two different examples of how you may handle errors are shown below.

Taking action within the error handler

In this example, the audio player will only move on to playing the next audio resource if an error has occurred. If playback ends gracefully, nothing will happen. This example avoids a transition into the Idle state.

const { createAudioResource } = require('@discordjs/voice');

const resource = createAudioResource('/home/user/voice/music.mp3', {
	metadata: {
		title: 'A good song!',
	},
});

player.play(resource);

player.on('error', error => {
	console.error(`Error: ${error.message} with resource ${error.resource.metadata.title}`);
	player.play(getNextResource());
});
1
2
3
4
5
6
7
8
9
10
11
12
13
14

Taking action within the Idle state

In this example, the error event is used only for logging purposes. The audio player will naturally transition into the Idle state, and then another resource is played. This has the advantage of working with streams that come to an end gracefully, and those that are interrupted by errors.

const { createAudioResource } = require('@discordjs/voice');

const resource = createAudioResource('/home/user/voice/music.mp3', {
	metadata: {
		title: 'A good song!',
	},
});

player.play(resource);

player.on('error', error => {
	console.error(error);
});

player.on(AudioPlayerStatus.Idle, () => {
	player.play(getNextResource());
});
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17