This project captures audio streams from various systems, transmits them over the network to another system as audio input for playback. The goal is to achieve audio stream sharing across the vast majority of operating systems.
This is aquawius' first moderately sized C++ project. I have high hopes for it, hence naming it aqua
after the first
half of my name. I also hope aqua
remains evergreen, just like the color this word represents.
Attention: this
readme.md
translate by DeepSeek AI, may some translation is wrong. Seereadme_zh.md
.
If you find this useful, please star the project. Issues are welcome, and pull requests would delight me!
The idea of audio sharing originated during my university years, but it remained shelved until now, finally taking shape—a tribute to my college life.
The project was initially inspired by audio-share, which implemented audio streaming to Android devices. I referenced parts of its code (especially the ASIO coroutine section) during the early stages. Special thanks to the audio-share project and its contributors.
The system consists of a server and client. In short: The server captures the system's audio stream and sends it over the network to one or more clients for playback.
Key features of aqua
:
- Cross-platform support: Versions for Windows and Linux with interoperability (e.g., capture on Linux and play on Windows, or vice versa).
- Low-latency audio transmission: Estimated under 20ms.
- GUI (Qt) and CLI support: No more wrestling with terminal windows.
- Network resilience: Client-side adaptive buffering to combat jitter.
- Automatic default device switching: Seamlessly switches capture/playback streams when default devices change.
- Visual volume indicator: Monitor audio capture/playback status.
Qt6 version. (Windows/Linux)
Console version. (-V to enable verbose mode)
Deploy the server on the sharing device and the client on the playback device. Both must have bidirectional network access.
- Run
aqua-server
, optionally binding to a specific IP (default: LAN IP) and port (default: 10120). - For cross-subnet use, bind to
0.0.0.0
.
- Run
aqua-client
, specifying the server's IP and RPC port (default: 10120). - Provide the client's local IP (must be reachable by the server) and port (default: random between 49152-65535).
- Qt GUI users: Check
Use custom settings
to specify IP/port.
In console version, you can use --help
or -h
to see usage.
You should now hear audio captured by the server.
Shared Components:
- gRPC services
- Network layer
- Capture/playback services
Client-Specific:
- Adaptive buffer
Server-Specific:
- Session management
- Linux PipeWire capture (via Sink, native stream routing)
- Windows WASAPI capture with stream routing (fixed format)
- IPv4 support
- Session management
- Linux PipeWire playback
- Windows WASAPI playback with stream routing (fixed format)
- IPv4 support
- Keep-alive mechanism (sync with server sessions)
- Adaptive buffering (network jitter mitigation)
Consider IPv6 support
In public network environments, NAT clients generally do not need to provide IP and port information. Current implementation is only tested on LAN and has design flaws (requires client IP/port). Partial server refactoring is needed, but LAN users are unaffected.
Previously only supported Linux PipeWire capture/playback. Windows capture support added; Windows playback not implemented (planned for next version)
PipeWire/WASAPI capture & playback now fully supported
Audio stream format was previously fixed (Windows required 48kHz/2ch/16bit to avoid crashes)
aqua
now supports dynamic stream configuration. Clients automatically track server format changes. OS limitations may prevent certain format switches (fallback to system-recommended formats).
- Qt version: Configure via menu bar
- CLI version: Set initial format via command-line args (see
--help
). Runtime format changes unavailable.
No GUI support (Qt6)
aqua-qt-server
andaqua-qt-client
released with full GUI support.Android client development pending (low priority due to overlap with
audio-share
and lack of Android expertise)Heavy dependency on gRPC communication. Timeout handling needs improvement – clients may wait excessively during network interruptions before disconnecting.
Grateful acknowledgment to these excellent libraries and their contributors.
The licensing model is under consideration. Stricter terms from dependencies will take precedence where applicable.