ipeesee
v0.1.0
Published
Electron IPC communication made easy and sexy.
Downloads
17
Maintainers
Readme
IPeeSee (Inter-process communication)
IPeeSee is a JS module built for the sake of easing the pain while working with native IPC implementation in Electron.
IPeeSee is written in Typescript, is fully documented and provides all the necessary types.
Compared to some other modules, IPeeSee preserves the interfaces on both main and renderer process and tries to accomplish one-way / two-way communication in the simplest way possible.
Besides communication, IPeeSee implements custom IPCError and IPCResponse. Those are implemented for the sole sake of seperating errors/responses from other communication protocols.
Install
npm i ipeesee --save
API
.send(channel, data, options)
Sends a message to a channel.
Parameters
channel (string) - A channel to send a message to
data (any) - Data to send (optional)
options (object) - (optional)
window (BrowserWindow) - An instance of browserWindow to send a message to. Only required when using with ipcMain
reply (boolean) - By default, we always wait for the response. Setting this to false will make the communication one-way only. In simple words, we send a message and don't care about the response
timeout (number) - Time (seconds) to wait before we manually resolve the reply
Resolves either an IPCError or IPCResponse with the following statusCodes:
- 204 - Resolved with no data in the response
- 200 - Resolved with the data in the response
- 408 - Resoled by manually timing out. This IPCResponse will also come with a message indicating the timeout and the channel we timed out at
Keep in mind that reply and timeout are mutually exclusive, meaning that setting timeout only makes sense when we are actually waiting for the reply. If we aren't (if reply is set to false or not passed at all), timeout will take no effect.
.reply(channel, callback)
Adds a listener that replies to the provided channel
Parameters
channel (string) - A channel to reply to
callback(data) (Function) - Receives a message with optional data and responds back to the channel. In order to reply back, it's mandatory to return from this function
Usage
IPeeSee interfaces are unified. This means that regardless if you are working with main or renderer process, methods and arguments are the same.
Constructor
IPeeSee constructor takes the process type (ipcMain, ipcRenderer) and an optional browserWindow that you want to send a message to.
If you don't pass the window (browserWindow) to the constructor, you will have to pass it to .send() each time you want to send a message from main to renderer process.
If your application only uses a single window, it's a good idea in that case to simply pass a window to the constructor and not worry about passing it while using .send().
const IPeeSee = require('ipeesee').default;
// or
import IPeeSee from 'ipeesee';
const ipc = new IPeeSee(ipcMain, yourWindow);
ipc.send('foo', { foo: 'bar' }).then(...).catch(...);
Or if you don't pass the window to the constructor, you will have to specify it in each .send (only when sending messages from main to renderer)
ipc.send('foo', { foo: 'bar' }, { window: yourBrowserWindow }).then(...).catch(...);
Send a message that expects no reply
ipc.send('foo', { foo: 'bar' }, { reply: false });
Manually timeout the reply after N number of seconds
ipc.send('foo', { foo: 'bar' }, { timeout: 30 }).then(...).catch(...);
Renderer
import { ipcRenderer } from 'electron';
import IPeeSee from 'ipeesee';
const ipc = new IPeeSee(ipcRenderer);
ipc
.send('foo', { foo: 'bar' })
.then(response => {
console.log(response);
})
.catch(error => console.error(error)); // { message: 'Cabooom! }
Main
import { ipcMain } from 'electron';
import IPeeSee from 'ipeesee';
const ipc = new IPeeSee(ipcMain);
ipc.reply('foo', data => {
return new Promise(resolve => {
console.log(data); // data from sender
resolve({ response: 'Foo is so Barish!' }); // respond back to sender
});
});
Communicating from main to renderer is done exactly the same way.
Working with errors
IPC .send(channel, response) doesn't provide way to return node-like response e.g. (error, response). For that reason, we have to work with a single object and make the best out of it.
To work around this limitation, IPeeSee will check if your response object has .error property on it and if so, will reject it as IPCError. This means that no matter what, you will always resolve, either an error or a valid response.
Example with returning an IPCError
ipc
.send('get-todos', { id: 1 })
.then(todo => {
console.log(todo);
})
.catch(error => {
console.log(error);
});
ipc.reply('get-todos', data => {
return new Promise(resolve => {
request(`https://jsonplaceholder.typicode.com/todos/${data.id}`, (error, response, body) {
if(error) {
resolve({ error })
} else {
resolve(response);
}
});
});
});
In case of an error, it's imperative that your response object has a property called .error. The error object itself can have any custom properties.
IPCError response object looks like this:
{
type: 'IPC_ERROR',
error: {} // error you previously resolved
}
TODO
- Add CI server and how to build section
- Unit tests