pearCast/app.js
Raven Scott b10b4f390d feat: Implement broadcaster-hosted TURN functionality over Hyperswarm. Remove the need for Google TURN/STUN servers.
- Modified code to enable the broadcaster to act as a TURN server using Hyperswarm connections.
- Replaced external STUN/TURN servers with custom signaling over Hyperswarm.
- Configured RTCPeerConnection with an empty ICE servers array to prevent reliance on external servers.
- Implemented custom signaling protocol for exchanging offers, answers, and ICE candidates over Hyperswarm connections.
- Updated broadcaster to relay media streams, effectively mimicking TURN server behavior within the Hyperswarm network.
- Adjusted peer connection setup for both broadcaster and listener to use Hyperswarm for signaling and data transport.
- Removed dependency on third-party servers; all communication now occurs within the peer-to-peer network.
- Updated UI elements and controls to reflect the changes.
- Ensured that peer count updates correctly with new connection handling.
2024-11-24 06:10:31 -05:00

411 lines
14 KiB
JavaScript

// app.js
import Hyperswarm from 'hyperswarm';
import crypto from 'hypercore-crypto';
import b4a from 'b4a';
let swarm;
let stationKey = crypto.randomBytes(32); // Default random key for the station
let currentDeviceId = null; // To store the selected audio input device ID
let isBroadcasting = false;
let localStream; // For broadcaster's audio stream
let peerConnections = {}; // Store WebRTC peer connections
let dataChannels = {}; // Store data channels for signaling
let conns = []; // Store Hyperswarm connections
document.addEventListener("DOMContentLoaded", () => {
console.log("DOM fully loaded and parsed");
document.getElementById('create-station').addEventListener('click', () => {
console.log("Create Station button clicked");
const createStationModal = new bootstrap.Modal(document.getElementById('createStationModal'));
createStationModal.show();
});
document.getElementById('generate-new-key').addEventListener('click', () => {
stationKey = crypto.randomBytes(32);
document.getElementById('existing-key').value = b4a.toString(stationKey, 'hex');
console.log("New station key generated");
});
document.getElementById('create-station-button').addEventListener('click', () => {
const existingKey = document.getElementById('existing-key').value.trim();
stationKey = existingKey ? b4a.from(existingKey, 'hex') : stationKey;
console.log("Creating station with key:", b4a.toString(stationKey, 'hex'));
setupStation(stationKey);
const createStationModal = bootstrap.Modal.getInstance(document.getElementById('createStationModal'));
createStationModal.hide();
});
document.getElementById('leave-stream').addEventListener('click', () => {
console.log("Leave Stream button clicked");
leaveStation();
});
document.getElementById('join-station-button').addEventListener('click', () => {
console.log("Join Station button clicked");
joinStation();
const joinModal = bootstrap.Modal.getInstance(document.getElementById('joinModal'));
joinModal.hide();
});
document.getElementById('apply-audio-source').addEventListener('click', applyAudioSource);
populateAudioInputSources();
});
async function populateAudioInputSources() {
try {
const devices = await navigator.mediaDevices.enumerateDevices();
const audioInputSelect = document.getElementById('audio-input-select');
audioInputSelect.innerHTML = '';
devices.forEach((device) => {
if (device.kind === 'audioinput') {
const option = document.createElement('option');
option.value = device.deviceId;
option.textContent = device.label || `Microphone ${audioInputSelect.length + 1}`;
audioInputSelect.appendChild(option);
}
});
currentDeviceId = audioInputSelect.value;
console.log("Audio input devices populated");
} catch (err) {
console.error("Error enumerating devices:", err);
}
}
async function applyAudioSource() {
const selectedDeviceId = document.getElementById('audio-input-select').value;
if (selectedDeviceId !== currentDeviceId) {
currentDeviceId = selectedDeviceId;
if (isBroadcasting) {
console.log("Applying new audio source:", selectedDeviceId);
try {
const newStream = await navigator.mediaDevices.getUserMedia({
audio: { deviceId: currentDeviceId ? { exact: currentDeviceId } : undefined },
});
console.log("New audio stream obtained");
// Replace tracks in existing peer connections
for (const remoteKey in peerConnections) {
const peerConnection = peerConnections[remoteKey];
const senders = peerConnection.getSenders();
for (const sender of senders) {
if (sender.track && sender.track.kind === 'audio') {
const newTrack = newStream.getAudioTracks()[0];
await sender.replaceTrack(newTrack);
console.log(`Replaced audio track for peer ${remoteKey}`);
}
}
}
// Stop the old audio tracks
if (localStream) {
localStream.getTracks().forEach((track) => track.stop());
console.log("Old audio tracks stopped");
}
// Update the localStream
localStream = newStream;
console.log("localStream updated with new audio stream");
} catch (err) {
console.error("Error applying new audio source:", err);
alert("Failed to apply new audio source. Please try again.");
}
}
}
}
function updatePeerCount() {
const peerCount = conns.length;
const stationInfoElement = document.getElementById('station-info');
if (isBroadcasting) {
stationInfoElement.textContent = `Station ID: ${b4a.toString(stationKey, 'hex')}\nConnected Peers: ${peerCount}`;
} else {
stationInfoElement.textContent = `Connected Peers: ${peerCount}`;
}
console.log(`Peer count updated: ${peerCount}`);
}
async function setupStation(key) {
try {
console.log("Setting up station...");
// Initialize Hyperswarm
swarm = new Hyperswarm();
swarm.join(key, { client: false, server: true });
// Get user media (audio input)
localStream = await navigator.mediaDevices.getUserMedia({
audio: { deviceId: currentDeviceId ? { exact: currentDeviceId } : undefined },
});
console.log("Local audio stream obtained");
isBroadcasting = true;
swarm.on('connection', (conn) => {
console.log("New connection established");
conns.push(conn);
updatePeerCount(); // Update peer count when a new connection is established
const remoteKey = conn.remotePublicKey.toString('hex');
// Use the Hyperswarm connection as a data channel for signaling
dataChannels[remoteKey] = conn;
// Set up WebRTC peer connection
setupBroadcasterPeerConnection(conn, remoteKey);
conn.on('close', () => {
console.log("Connection closed with peer");
if (peerConnections[remoteKey]) {
peerConnections[remoteKey].close();
delete peerConnections[remoteKey];
}
delete dataChannels[remoteKey];
conns.splice(conns.indexOf(conn), 1);
updatePeerCount(); // Update peer count when a connection is closed
});
conn.on('error', (err) => {
console.error("Connection error with peer:", err);
if (peerConnections[remoteKey]) {
peerConnections[remoteKey].close();
delete peerConnections[remoteKey];
}
delete dataChannels[remoteKey];
conns.splice(conns.indexOf(conn), 1);
updatePeerCount(); // Update peer count on error
});
});
document.getElementById('broadcaster-controls').classList.remove('d-none');
document.getElementById('setup').classList.add('d-none');
document.getElementById('controls').classList.remove('d-none');
document.getElementById('station-info').textContent = `Station ID: ${b4a.toString(key, 'hex')}`;
console.log("Station setup complete");
} catch (err) {
console.error("Error setting up station:", err);
alert("Failed to set up station. Please try again.");
}
}
function setupBroadcasterPeerConnection(conn, remoteKey) {
const configuration = {
iceServers: [], // Empty array since we are not using external STUN/TURN servers
};
const peerConnection = new RTCPeerConnection(configuration);
peerConnections[remoteKey] = peerConnection;
// Add local stream tracks to peer connection
localStream.getTracks().forEach((track) => {
peerConnection.addTrack(track, localStream);
console.log("Added track to peer connection:", track);
});
// Handle ICE candidates
peerConnection.onicecandidate = ({ candidate }) => {
if (candidate) {
console.log("Sending ICE candidate to peer");
conn.write(JSON.stringify({ type: 'candidate', candidate }));
}
};
peerConnection.oniceconnectionstatechange = () => {
console.log("Broadcaster ICE connection state changed to:", peerConnection.iceConnectionState);
};
// Handle incoming signaling data
conn.on('data', async (data) => {
const message = JSON.parse(data.toString());
await handleBroadcasterSignalingData(conn, message, remoteKey);
});
}
async function handleBroadcasterSignalingData(conn, message, remoteKey) {
const peerConnection = peerConnections[remoteKey];
if (message.type === 'offer') {
console.log("Received offer from peer");
await peerConnection.setRemoteDescription(new RTCSessionDescription(message.offer));
console.log("Set remote description with offer from peer");
const answer = await peerConnection.createAnswer();
await peerConnection.setLocalDescription(answer);
console.log("Created and set local description with answer");
// Send the answer back to the listener
conn.write(JSON.stringify({ type: 'answer', answer }));
console.log("Sent answer to peer");
} else if (message.type === 'candidate') {
console.log("Received ICE candidate from peer");
await peerConnection.addIceCandidate(new RTCIceCandidate(message.candidate));
}
}
async function joinStation() {
try {
const stationId = document.getElementById('station-id').value;
if (!stationId) {
alert("Please enter a station ID.");
return;
}
console.log("Joining station with ID:", stationId);
const topicBuffer = b4a.from(stationId, 'hex');
swarm = new Hyperswarm();
swarm.join(topicBuffer, { client: true, server: false });
swarm.on('connection', (conn) => {
console.log("Connected to broadcaster");
conns.push(conn);
updatePeerCount(); // Update peer count when a new connection is established
const remoteKey = conn.remotePublicKey.toString('hex');
// Use the Hyperswarm connection as a data channel for signaling
dataChannels[remoteKey] = conn;
// Set up WebRTC peer connection
setupListenerPeerConnection(conn, remoteKey);
conn.on('close', () => {
console.log("Connection closed with broadcaster");
if (peerConnections[remoteKey]) {
peerConnections[remoteKey].close();
delete peerConnections[remoteKey];
}
delete dataChannels[remoteKey];
conns.splice(conns.indexOf(conn), 1);
updatePeerCount(); // Update peer count when a connection is closed
});
conn.on('error', (err) => {
console.error("Connection error with broadcaster:", err);
if (peerConnections[remoteKey]) {
peerConnections[remoteKey].close();
delete peerConnections[remoteKey];
}
delete dataChannels[remoteKey];
conns.splice(conns.indexOf(conn), 1);
updatePeerCount(); // Update peer count on error
});
updatePeerCount();
});
document.getElementById('station-info').textContent = `Connected to Station: ${stationId}`;
document.getElementById('setup').classList.add('d-none');
document.getElementById('controls').classList.remove('d-none');
document.getElementById('listener-controls').classList.remove('d-none');
console.log("Joined station successfully");
} catch (err) {
console.error("Error joining station:", err);
alert("Failed to join station. Please try again.");
}
}
function setupListenerPeerConnection(conn, remoteKey) {
const configuration = {
iceServers: [], // Empty array since we are not using external STUN/TURN servers
};
const peerConnection = new RTCPeerConnection(configuration);
peerConnections[remoteKey] = peerConnection;
// Handle incoming tracks (audio streams)
peerConnection.ontrack = (event) => {
console.log("Received remote track");
const [remoteStream] = event.streams;
// Play the remote audio stream
const audioElement = document.createElement('audio');
audioElement.srcObject = remoteStream;
audioElement.autoplay = true;
document.body.appendChild(audioElement);
console.log("Audio element created and playback started");
};
// Handle ICE candidates
peerConnection.onicecandidate = ({ candidate }) => {
if (candidate) {
console.log("Sending ICE candidate to broadcaster");
conn.write(JSON.stringify({ type: 'candidate', candidate }));
}
};
peerConnection.oniceconnectionstatechange = () => {
console.log("Listener ICE connection state changed to:", peerConnection.iceConnectionState);
};
// Handle signaling data from broadcaster
conn.on('data', async (data) => {
const message = JSON.parse(data.toString());
await handleListenerSignalingData(conn, message, remoteKey);
});
initiateOffer(conn, peerConnection);
}
async function handleListenerSignalingData(conn, message, remoteKey) {
const peerConnection = peerConnections[remoteKey];
if (message.type === 'answer') {
console.log("Received answer from broadcaster");
await peerConnection.setRemoteDescription(new RTCSessionDescription(message.answer));
console.log("Set remote description with answer from broadcaster");
} else if (message.type === 'candidate') {
console.log("Received ICE candidate from broadcaster");
await peerConnection.addIceCandidate(new RTCIceCandidate(message.candidate));
}
}
async function initiateOffer(conn, peerConnection) {
try {
console.log("Initiating offer to broadcaster");
// Add transceiver to receive audio
peerConnection.addTransceiver('audio', { direction: 'recvonly' });
const offer = await peerConnection.createOffer();
await peerConnection.setLocalDescription(offer);
console.log("Created and set local description with offer");
// Send offer to broadcaster
conn.write(JSON.stringify({ type: 'offer', offer }));
console.log("Sent offer to broadcaster");
} catch (err) {
console.error("Error initiating offer:", err);
}
}
function leaveStation() {
console.log("Leaving station...");
if (swarm) {
swarm.destroy();
console.log("Swarm destroyed");
}
// Close all peer connections
Object.values(peerConnections).forEach((pc) => pc.close());
peerConnections = {};
// Close all Hyperswarm connections
conns.forEach((conn) => conn.destroy());
conns = [];
// Stop local media tracks
if (localStream) {
localStream.getTracks().forEach((track) => track.stop());
localStream = null;
console.log("Local media tracks stopped");
}
isBroadcasting = false;
document.getElementById('setup').classList.remove('d-none');
document.getElementById('controls').classList.add('d-none');
document.getElementById('broadcaster-controls').classList.add('d-none');
document.getElementById('listener-controls').classList.add('d-none');
document.getElementById('station-info').textContent = '';
console.log("Left the station.");
}