pagraphcontrol/store/pulse-middleware.js

261 lines
7.7 KiB
JavaScript

const Bluebird = require('bluebird');
const PAClient = require('@futpib/paclient');
const { handleActions } = require('redux-actions');
const { pulse: pulseActions } = require('../actions');
const { things } = require('../constants/pulse');
const { getPaiByTypeAndIndex } = require('../selectors');
function getFnFromType(type) {
let fn;
switch (type) {
case 'sink':
case 'card':
case 'source':
fn = type;
break;
case 'sinkInput':
case 'sourceOutput':
case 'client':
case 'module':
fn = `${type}ByIndex`;
break;
default:
throw new Error('Unexpected type: ' + type);
}
return 'get' + fn[0].toUpperCase() + fn.slice(1);
}
function setSinkChannelVolume(pa, store, index, channelIndex, volume, cb) {
const pai = getPaiByTypeAndIndex('sink', index)(store.getState());
pa.setSinkVolumes(index, pai.channelVolumes.map((v, i) => i === channelIndex ? volume : v), cb);
}
function setSourceChannelVolume(pa, store, index, channelIndex, volume, cb) {
const pai = getPaiByTypeAndIndex('source', index)(store.getState());
pa.setSourceVolumes(index, pai.channelVolumes.map((v, i) => i === channelIndex ? volume : v), cb);
}
function setSinkInputChannelVolume(pa, store, index, channelIndex, volume, cb) {
const pai = getPaiByTypeAndIndex('sinkInput', index)(store.getState());
pa.setSinkInputVolumesByIndex(index, pai.channelVolumes.map((v, i) => i === channelIndex ? volume : v), cb);
}
function setSourceOutputChannelVolume(pa, store, index, channelIndex, volume, cb) {
const pai = getPaiByTypeAndIndex('sourceOutput', index)(store.getState());
pa.setSourceOutputVolumesByIndex(index, pai.channelVolumes.map((v, i) => i === channelIndex ? volume : v), cb);
}
module.exports = store => {
const pa = new PAClient();
const getInfo = (type, index) => {
let method;
try {
method = getFnFromType(type);
} catch (error) {
if (error.message.startsWith('Unexpected type:')) {
console.warn(error);
return;
}
throw error;
}
pa[method](index, (err, info) => {
if (err) {
if (err.message === 'No such entity') {
console.warn(err.message, type, index);
return;
}
throw err;
}
info.type = info.type || type;
store.dispatch(pulseActions.info(info));
});
};
pa
.on('ready', () => {
store.dispatch(pulseActions.ready());
pa.subscribe('all');
getServerInfo();
things.forEach(({ method, type }) => {
pa[method]((err, infos) => {
handleError(err);
infos.forEach(info => {
const { index } = info;
info.type = info.type || type;
store.dispatch(pulseActions.new({ type, index }));
store.dispatch(pulseActions.info(info));
});
});
});
})
.on('close', () => {
store.dispatch(pulseActions.close());
reconnect();
})
.on('new', (type, index) => {
if (type === 'server') {
getServerInfo();
return;
}
store.dispatch(pulseActions.new({ type, index }));
getInfo(type, index);
})
.on('change', (type, index) => {
if (type === 'server') {
getServerInfo();
return;
}
store.dispatch(pulseActions.change({ type, index }));
getInfo(type, index);
})
.on('remove', (type, index) => {
store.dispatch(pulseActions.remove({ type, index }));
})
.on('error', error => {
console.error(error);
});
const reconnect = () => new Bluebird((resolve, reject) => {
pa.once('ready', resolve);
pa.once('error', reject);
pa.connect();
}).catch(error => {
if (error.message === 'Unable to connect to PulseAudio server') {
return Bluebird.delay(5000).then(reconnect);
}
throw error;
});
reconnect();
const getServerInfo = () => {
pa.getServerInfo((err, info) => {
if (err) {
handleError(err);
} else {
store.dispatch(pulseActions.serverInfo(info));
}
});
};
const handleError = error => {
if (!error) {
return;
}
console.error(error);
store.dispatch(pulseActions.error(error));
};
const handlePulseActions = handleActions({
[pulseActions.moveSinkInput]: (state, { payload: { sinkInputIndex, destSinkIndex } }) => {
pa.moveSinkInput(sinkInputIndex, destSinkIndex, handleError);
return state;
},
[pulseActions.moveSourceOutput]: (state, { payload: { sourceOutputIndex, destSourceIndex } }) => {
pa.moveSourceOutput(sourceOutputIndex, destSourceIndex, handleError);
return state;
},
[pulseActions.killClientByIndex]: (state, { payload: { clientIndex } }) => {
pa.killClientByIndex(clientIndex, handleError);
return state;
},
[pulseActions.killSinkInputByIndex]: (state, { payload: { sinkInputIndex } }) => {
pa.killSinkInputByIndex(sinkInputIndex, handleError);
return state;
},
[pulseActions.killSourceOutputByIndex]: (state, { payload: { sourceOutputIndex } }) => {
pa.killSourceOutputByIndex(sourceOutputIndex, handleError);
return state;
},
[pulseActions.loadModule]: (state, { payload: { name, argument } }) => {
pa.loadModule(name, argument, handleError);
return state;
},
[pulseActions.unloadModuleByIndex]: (state, { payload: { moduleIndex } }) => {
pa.unloadModuleByIndex(moduleIndex, handleError);
return state;
},
[pulseActions.setSinkVolumes]: (state, { payload: { index, channelVolumes } }) => {
pa.setSinkVolumes(index, channelVolumes, handleError);
return state;
},
[pulseActions.setSourceVolumes]: (state, { payload: { index, channelVolumes } }) => {
pa.setSourceVolumes(index, channelVolumes, handleError);
return state;
},
[pulseActions.setSinkInputVolumes]: (state, { payload: { index, channelVolumes } }) => {
pa.setSinkInputVolumesByIndex(index, channelVolumes, handleError);
return state;
},
[pulseActions.setSourceOutputVolumes]: (state, { payload: { index, channelVolumes } }) => {
pa.setSourceOutputVolumesByIndex(index, channelVolumes, handleError);
return state;
},
[pulseActions.setSinkChannelVolume]: (state, { payload: { index, channelIndex, volume } }) => {
return setSinkChannelVolume(pa, store, index, channelIndex, volume, handleError);
},
[pulseActions.setSourceChannelVolume]: (state, { payload: { index, channelIndex, volume } }) => {
return setSourceChannelVolume(pa, store, index, channelIndex, volume, handleError);
},
[pulseActions.setSinkInputChannelVolume]: (state, { payload: { index, channelIndex, volume } }) => {
return setSinkInputChannelVolume(pa, store, index, channelIndex, volume, handleError);
},
[pulseActions.setSourceOutputChannelVolume]: (state, { payload: { index, channelIndex, volume } }) => {
return setSourceOutputChannelVolume(pa, store, index, channelIndex, volume, handleError);
},
[pulseActions.setCardProfile]: (state, { payload: { index, profileName } }) => {
pa.setCardProfile(index, profileName, handleError);
return state;
},
[pulseActions.setSinkMute]: (state, { payload: { index, muted } }) => {
pa.setSinkMute(index, muted, handleError);
return state;
},
[pulseActions.setSourceMute]: (state, { payload: { index, muted } }) => {
pa.setSourceMute(index, muted, handleError);
return state;
},
[pulseActions.setSinkInputMuteByIndex]: (state, { payload: { index, muted } }) => {
pa.setSinkInputMuteByIndex(index, muted, handleError);
return state;
},
[pulseActions.setSourceOutputMuteByIndex]: (state, { payload: { index, muted } }) => {
pa.setSourceOutputMuteByIndex(index, muted, handleError);
return state;
},
[pulseActions.setDefaultSinkByName]: (state, { payload: { name } }) => {
pa.setDefaultSinkByName(name, handleError);
return state;
},
[pulseActions.setDefaultSourceByName]: (state, { payload: { name } }) => {
pa.setDefaultSourceByName(name, handleError);
return state;
},
}, null);
return next => action => {
const ret = next(action);
handlePulseActions(null, action);
return ret;
};
};