Summary of what I am trying to achieve:
I'm currently doing some work on a Discord bot. I'm trying to join a voice channel, which is the easy part, and then use the combined audio of the speakers in that voice channel as input for a webpage in a web browser. It doesn't really matter which browser it is as long as it can be controlled with Selenium.
My bot so far is written up in Python using the discord.py API wrapper. Unfortunately listening to, as opposed to putting in, audio hasn't been exactly implemented great − let alone documented − with discord.py. This made me decide to switch to node.js (i.e. discord.js) for the voice channel stuff of my bot.
After switching to discord.js it was pretty easy to determine who's talking and create an audio stream (PCM stream) for that user. For the next part I though I'd just pipe the audio stream to a virtual microphone and select that as the audio input on the browser. You can even use FFMPEG from within node.js 1, to get something that looks like this:
const Discord = require("discord.js");
const client = new Discord.Client();
client.on('ready', () => {
voiceChannel = client.channels.get('SOME_CHANNEL_ID');
voiceChannel.join()
.then(conn => {
console.log('Connected')
const receiver = conn.createReceiver();
conn.on('speaking', (user, speaking) => {
if (speaking) {
const audioStream = receiver.createPCMStream(user);
ffmpeg(stream)
.inputFormat('s32le')
.audioFrequency(16000)
.audioChannels(1)
.audioCodec('pcm_s16le')
.format('s16le')
.pipe(someVirtualMic);
}
});
})
.catch(console.log);
});
client.login('SOME_TOKEN');
This last part, creating and streaming to a virtual microphone, has proven to be rather complicated. I've read a ton of SO posts and documentation on both The Advanced Linux Sound Architecture (ALSA) and the JACK Audio Connection Kit, but I simply can't figure out how to setup a virtual microphone that will show up as a mic in my browser, or how to pipe audio to it.
Any help or pointers to a solution would be greatly appreciated!
For the past couple of days I've kept on looking into to this issue. I've now learned about ALSA loopback devices and feel that the solution must be there.
I've pretty much followed a post that talks about loopback devices and aims to achieve the following:
Simply imagine that you have a physical link between one OUT and one IN of the same device.
I've set up the devices as described in the post and now two new audio devices show up when selecting a microphone in Firefox. I'd expect one, but I that may be because I don't entirely understand the loopback devices (yet).
The loop back devices are created and I think that they're linked (if I understood the aforementioned article correctly). Assuming that's the case the only problem I have to tackle is streaming the audio via FFMPEG from within node.js.
This virtual device effectively transfers the audio from the video playing in the background to a virtual microphone input. You can now select this “microphone” in any app, or set it as default for all apps. The best part is that your standard audio is unaffected by this virtual device and you can use your real microphone whenever you want.
The Icons can be clicked on and VirtualDJ will configure the audio with the most common settings automatically in the advanced menu. From there, audio configurations can be altered added and removed depending on the audio interface’s capabilities and the requirements of the user.
A microphone may be routed to the Master Output, using an input of a multi-channel sound card by simply connecting a microphone to the input or Microphone Jack (if equipped). Select MICROPHONE from the INPUT options in order for an additional line to be created. This can also be done by clicking on button under Inputs in the advanced menu.
A microphone may be routed to the Master Output, using an input of a multi-channel sound card by simply connecting a microphone to the input or Microphone Jack (if equipped). Select MICROPHONE from the INPUT options in order for an additional line to be created.
This is answered there : Linux pipe audio file to microphone input
Create a virtual microphone :
pactl load-module module-pipe-source source_name=virtmic file=/tmp/virtmic format=s16le rate=16000 channels=1
Pipe ffmpeg output to the virtmic
file and it should work :
ffmpeg -re \
-i input.mp3 \
-f s16le -ar 16000 -ac 1 - > /tmp/virtmic
Note : I have noticed that if there are no readers to the pipe, ffmpeg
just hangs. This can be fixed by simply opening Pulse Audio VU meter, command pavucontrol
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With