Node.js module to record PCM audio data from ALSA capture devices (e.g., microphones).
Emits events about overruns, short reads, deviating sample rates or period sizes.
Tested (module compiles and records audio) with the following Node versions:
v6.17.1 (lts/boron)
v8.9.4
v8.11.3
v8.17.0 (lts/carbon)
v10.22.1 (lts/dubnium)
v12.14.1
v12.18.4 (lts/erbium)
v14.16.0 (lts/fermium)
v15.11.0
v16.13.0 (lts/gallium)
v17.1.0
v18.11.0
(successfully built, but audio recording not tested)v19.0.0 (current)
To build native Node.js modules, this projects uses node-gyp, which requires[1]:
- Python v3.6, v3.7, v3.8, or v3.9
- make
- A proper C/C++ compiler toolchain, like GCC (must support 'c++17') )
This module also requires a libasound2-dev
for <alsa/asoundlib.h>
and libasound2
to link against libasound.so.2
.
On debian derivatives system you can install all dependencies with apt
:
sudo apt install libasound2 libasound2-dev
npm install alsa-capture
or
yarn add alsa-capture
Run npm install
, which will execute a node-gyp rebuild
and build./build/Release/capture.node
. The capture.node
file is needed if you want to distribute the package in binary form (libasound.so.2
is still needed).
const AlsaCapture = require("alsa-capture");
// default values for the options object
const captureInstance = new AlsaCapture({
channels: 2,
debug: false,
device: "default",
format: "S16_LE",
periodSize: 32,
periodTime: undefined,
rate: 44100,
});
// data is an Uint8Array
// Buffer size = numChannels * formatByteSize * periodSize
// Example: 2 Bytes (AlsaFormat.S16_LE) * 2 (numChannels) * 32 (periodSize) = 128 Bytes
captureInstance.on("audio", (data) => {
console.log(data);
});
// if the requested rate is not available for the capture device
// ALSA will select the nearest available
captureInstance.on("rateDeviating", (actualRate) => {
console.warn(`Sound card rate deviates from requested rate; Actual rate: ${actualRate}`);
});
// if the requested period size is not available for the capture device
// ALSA will select the nearest available
captureInstance.on("periodSizeDeviating", (actualPeriodSize) => {
console.warn(`Sound card period size deviates from requested period size; Actual period size: ${actualPeriodSize}`);
});
captureInstance.on("periodTime", (periodTime) => {
console.log(`Actual period time ${periodTime}`);
});
// if an error occurs, error will contain the message
captureInstance.on("error", (error) => {
console.error(error);
});
// if overruns occur frequently try increasing the period_size
captureInstance.on("overrun", () => {
console.warn("NodeAlsaCapture overrun");
});
captureInstance.on("shortRead", (framesRead) => {
console.warn(`NodeAlsaCapture shortRead: Only ${framesRead} read`);
});
captureInstance.on("readError", (error) => {
console.warn(`NodeAlsaCapture readError: ${error}`);
});
setTimeout(() => {
// close capture device
captureInstance.close();
}, 10000);
// event is sent after capture devices is closed completely
captureInstance.on("close", () => {
console.log("capture closed");
});
Creates a new ALSA capture instance which extends eventemitter3 and emits several kinds of events including the PCM audio data. The optional opts
object sets the ALSA hardware parameters.
option | type | description | default |
---|---|---|---|
channels | number | select number of channels to capture | 2 |
debug | boolean | prints debug data to stderr | false |
device | string | ALSA device ID | default |
format | string | Sample format (see Supported Sample formats) | S16_LE |
periodSize | number | A period is the number of frames in between each hardware interrupt | 32 |
periodTime | number | Set period time near n us. | (no default) |
rate | number | Sample rate (400 <= rate <= 196000) | 44100 |
Note: snd_pcm_hw_params_set_period_time_near
will only be called if the opts
object has the periodTime
property.
Note: snd_pcm_hw_params_set_access is set to SND_PCM_ACCESS_RW_INTERLEAVED.
Stops the ALSA capture thread. Afterwards the close
event will be emitted.
Returns the PCM data in an Uint8Array
.
The buffer size is derived from the number of channels, the sample format and the period size:
bufferSize = numChannels * formatByteSize * periodSize
Capture instance closed.
If the requested sample rate is not available for the capture device ALSA will select the nearest available
If the requested period size is not available for the capture device ALSA will select the nearest available
The actual period time
Exceptions in Instance creation. See error
messages.
An ALSA Buffer overrun occurred.
ALSA could not capture enough frames, only framesRead
frames.
Read error message. See snd_strerror
.
arecord -l
lists all soundcards and digital audio devices:
card 0: PCH [HDA Intel PCH], device 0: CX20590 Analog [CX20590 Analog]
Subdevices: 1/1
Subdevice #0: subdevice #0
card 1: CODEC [USB Audio CODEC], device 0: USB Audio [USB Audio]
Subdevices: 1/1
Subdevice #0: subdevice #0
To select card 1 use hw:1,0
or if you want to enable the ALSA plugin layer (ALSA will automatically convert the data to a format supported by your sound card 1) use plughw:1,0
.
arecord -L
lists all list device names
default
Playback/recording through the PulseAudio sound server
null
Discard all samples (playback) or generate zero samples (capture)
pulse
PulseAudio Sound Server
sysdefault:CARD=PCH
HDA Intel PCH, CX20590 Analog
Default Audio Device
front:CARD=PCH,DEV=0
HDA Intel PCH, CX20590 Analog
Front speakers
hw:CARD=PCH,DEV=0
HDA Intel PCH, CX20590 Analog
Direct hardware device without any conversions
plughw:CARD=PCH,DEV=0
HDA Intel PCH, CX20590 Analog
hw:CARD=CODEC,DEV=0
USB Audio CODEC, USB Audio
Direct hardware device without any conversions
plughw:CARD=CODEC,DEV=0
USB Audio CODEC, USB Audio
Hardware device with all software conversions
So you could also use hw:CARD=CODEC,DEV=0
to write to this external USB audio card or use pulse
and use pulse audio settings.
- S8
- U8
- S16_LE
- S16_BE
- U16_LE
- U16_BE
- S24_LE
- S24_BE
- U24_LE
- U24_BE
- S32_LE
- S32_BE
- U32_LE
- U32_BE
- FLOAT_LE
- FLOAT_BE
- FLOAT64_LE
- FLOAT64_BE
- IEC958_SUBFRAME_LE
- IEC958_SUBFRAME_BE
- MU_LAW
- A_LAW
- IMA_ADPCM
- MPEG
- GSM
- SPECIAL
- S24_3LE
- S24_3BE
- U24_3LE
- U24_3BE
- S20_3LE
- S20_3BE
- U20_3LE
- U20_3BE
- S18_3LE
- S18_3BE
- U18_3LE
- U18_3BE
- G723_24
- G723_24_1B
- G723_40
- G723_40_1B
- DSD_U8
- DSD_U16_LE
- DSD_U32_LE
- DSD_U16_BE
MIT
Copyright (c) 2020, 2021, 2022 Bernd Kaiser
- The ALSA c code examples
- The streaming worker was originally written by Scott Frees.