---
layout: post
title: "在 Electron 使用 IPC 串聯前端和 Node API"
date: 2016-11-02
---
# 在 Electron 使用 IPC 串聯前端和 Node API
這是我最近在實作噗浪 electron app - [**Puraku**][puraku] 時,使用的抽象化寫法。
先談一下背景。其實在官方的 [Plurk API][plurk-api] 頁面上就已經有 [JavaScript 的噗浪 API Library][purakujs]了,不過它沒有包成 npm 可以直接使用,而且還相依於 [node-oauth][node-oauth] 套件,看名字就知道和 Node 有關。
這次寫的 [puraku][puraku] 是一個以前端為主的桌面軟體,所以我勢必要對噗浪的 API 套件做些改寫。
**更新**: 其實 Electron renderer process 就能呼叫 node API 了,只是我先入為主的以為 main process 才能使用,所以引發了這個軟體問題 XDrz。
## 重新封裝 API Library
我已經包裝成 [purakujs][purakujs],它是個可以直接使用的 Plurk API Node library,雖然這年頭也沒多少工程師在串噗浪 API 了 😅 。值得一提的是 yarn 對 `npm link` 的支援不太好,在設定本機開發環境跑完 `npm link` 後,不要在 `package.json` 修改套件版本,防止在跑 `yarn install` 指令時又噴錯。
### Electron
關於 Electron 是啥便不再贅述。~~要知道的是只有在 Electron 的 Main Process 裡才可以呼叫 node 的 api~~(錯了),所以把 node-oauth 套件放在這跑是沒問題的,但 Renderer Process 才是主要觸發 API 的地方(換頁、捲動、按鈕等)。Electron 提供了 IPC 的 API 界面實作,可以這樣寫:
> commit [df4c8a2](https://github.com/puraku/client/pull/5/commits/df4c8a225a36485747f7022b1391b50ee9e9f19c)
```javascript
// Renderer Process
import { ipcRenderer } from 'electron';
export function request(method, endpoint, params=null) {
return ipcRenderer.sendSync('puraku:api', {method, endpoint, params});
}
// Main Process
ipcMain.on('puraku:api', (event, args) => {
const { method, endpoint, params } = args;
myApiClient.request(method, endpoint, params).then(({data}) => {
event.returnValue = JSON.parse(data);
}).catch(error => {
event.returnValue = { error };
});
});
```
Renderer Process 送出 API 請求到 Main Process,已經正在監聽的 Main Process 在呼叫完 API 請求(`myAPIClient.request`)之後,返還資料給 Renderer Process。在這裡透過 ipcMain 建立叫做 `puraku:api` 的事件監聽,由 `ipcRenderer` 送出事件請求。`event.returnValue` 是 ipc 的同步寫法。一旦使用同步,整個 Renderer Process 在送出事件請求(`ipcRenderer.sendSync`)之後會被 Block,所以我改用非同步的 IPC 寫法,再用 Promise 封裝。
### Asyncronous IPC
> commit [3f2fee](https://github.com/puraku/client/pull/5/commits/3f2fee64664c4082520a3a8b9ffe6d90cb6cfdbd)
```javascript
// Renderer Process
import { ipcRenderer } from 'electron';
export function request(method, endpoint, params=null) {
return new Promise((resolve, reject) => {
const timestamp = Date.now();
const result = ipcRenderer.send('puraku:api', {method, endpoint, params, timestamp});
ipcRenderer.once(`puraku:api:${endpoint}:${timestamp}`, (event, result) => {
if (result.hasOwnProperty('error')) {
reject(result);
} else {
resolve(result);
}
});
});
}
// Main Process
ipcMain.on('puraku:api', (event, args) => {
const { method, endpoint, params, timestamp } = args;
myApiClient.request(method, endpoint, params).then(({data}) => {
event.sender.send(`puraku:api:${endpoint}:${timestamp}`, JSON.parse(data));
}).catch(error => {
event.sender.send(`puraku:api:${endpoint}:${timestamp}`, {error});
});
});
```
這一版跟上面的差別在於,Renderer Process 送請求給 Main Process 之後,馬上建立另一個 IPC 的 Listener 等待 Main Process 回調,而 Main Process 在處裡完 API 請求之後(`myAPIClient.request`) 再用 IPC 非同步寫法回傳資料(`event.sender.send`)。在這個版本 IPC 關係變得比較複雜,簡單畫了一下:
```txt
Promise start +-----+
| Renerer Process Main Process
|
+-----> +--------------------+
| |
| ipcRenderer.send |
| | +---------------------+
+------------------------------+ |
| | | API request |
| ipcRenderer.once | | |
| | | event.sender.send |
| create listener | | |
| | | |
+---------+----------+ +-----------+---------+
| |
| |
| |
+---------+----------+ |
| <---------------------+
| Event received |
| |
+-----+ +---------+----------+
| |
| |
Promise end <-----+ |
|
|
|
|
|
|
|
|
v
```
### IPC 事件一對一對應
可以發現在這一版的 Renderer Process 我加了一個 `timestamp` 來簡單的區分不同的 API request,因為如果光用 API 的 Endpoint 當做事件的鍵值,戳相同 API 兩次時就會衝到。
```javascript
const timestamp = Date.now();
const eventKey = `puraku:api:${endpoint}:${timestamp}`;
```
不過卻發現每次取的 timestamp 還是有機會一樣,經過 Google 之後我把 `timestamp` 亂數的產生方法改成 `performance.now()`:
```javascript
const randomSeed = performance.now();
const eventKey = `puraku:api:${endpoint}:${randomSeed}`;
```
到這裡,我們就可以用熟悉的 Promise 介面,在 Renderer Process 輕鬆地串接 Main Process 的 API 啦!以下是目前的實作:
```javascript
// Renderer Process
import { ipcRenderer } from 'electron';
export function request(method, endpoint, params = null) {
return new Promise((resolve, reject) => {
const randomSeed = performance.now();
ipcRenderer.send('puraku:api', { method, endpoint, params, randomSeed });
ipcRenderer.once(`puraku:api:${endpoint}:${randomSeed}`, (event, result) => {
if (result.hasOwnProperty('error')) {
reject(result);
} else {
resolve(result);
}
});
});
}
// Main Process
const { ipcMain } = require('electron');
ipcMain.on('puraku:api', (event, args) => {
const { method, endpoint, params, randomSeed } = args;
myApiClient.request(method, endpoint, params).then(({data}) => {
event.sender.send(`puraku:api:${endpoint}:${randomSeed}`, JSON.parse(data));
}).catch(error => {
event.sender.send(`puraku:api:${endpoint}:${randomSeed}`, {error});
});
});
```
在 Renderer Process 裡(在我的例子裡是 Vue 前端 App)就可以用簡單的介面來呼叫 API 啦!
## 其它
聽說用 message queue 來實作比較好,不過 It works for now,就暫時沒有更新實作的動力(懶)
可以在 [puraku/client PR#5](https://github.com/puraku/client/pull/5) 閱讀實作的過程。自從對 Redmine 上癮之後,連 GitHub Flow 也一併愛上了,在本 Repo 一個功能就開 Branch 做成 Pull Request,就算一個人的 GitHub Flow 也能玩的愉悅 XD
[plurk-api]: https://www.plurk.com/API
[puraku]: https://github.com/puraku/client
[purakujs]: https://github.com/puraku/purakujs
[plurkjs]: https://github.com/clsung/plurkjs
[node-oauth]: https://github.com/ciaranj/node-oauth