const { app, BrowserWindow, ipcMain, shell, clipboard } = require('electron') const Path = require('node:path') const URL = require('node:url') const p = require('node:util').promisify const Keypair = require('ppppp-keypair') const awaitable = require('pull-awaitable') // WARNING monkey patch! ------------------------------------------------------- const na = require('sodium-native') na.sodium_malloc = function sodium_malloc_monkey_patched(n) { return Buffer.alloc(n) } na.sodium_free = function sodium_free_monkey_patched() {} // Electron > 20.3.8 breaks a napi method that `sodium_malloc` depends on to // create external buffers. (see v8 memory cage) // // This crashes electron when called by various libraries, so we monkey-patch // this particular function. // ----------------------------------------------------------------------------- process.env.ZOOBOARD_DATA ??= Path.join(app.getPath('appData'), 'zooboard') app.setPath('userData', process.env.ZOOBOARD_DATA) const path = process.env.PZP_DIR ?? Path.resolve(app.getPath('userData'), 'ppppp') const keypairPath = Path.join(path, 'keypair.json') const keypair = Keypair.loadOrCreateSync(keypairPath) let mainWindow let globalAccountID = null let globalAccountName = null const peer = require('secret-stack/bare')() .use(require('secret-stack/plugins/net')) .use(require('secret-handshake-ext/secret-stack')) .use(require('ppppp-net')) .use(require('ppppp-db')) .use(require('ppppp-set')) .use(require('ppppp-dict')) .use(require('ppppp-goals')) .use(require('ppppp-sync')) .use(require('ppppp-gc')) .use(require('ppppp-conductor')) .use(require('ppppp-hub-client')) .use(require('ppppp-promise')) .use(require('ppppp-invite')) .call(null, { shse: { caps: require('ppppp-caps'), }, global: { keypair, path, timers: { inactivity: 10 * 60e3, }, connections: { incoming: { tunnel: [{ transform: 'shse', scope: 'public' }], }, outgoing: { net: [{ transform: 'shse' }], tunnel: [{ transform: 'shse' }], }, }, }, net: { autostart: false, }, }) function createWindow() { mainWindow = new BrowserWindow({ width: 1200, height: 1200, title: 'Zooboard', webPreferences: { preload: Path.join(__dirname, 'preload.js'), }, }) const startUrl = process.env.ELECTRON_START_URL ?? URL.format({ pathname: Path.join(__dirname, '/../build/index.html'), protocol: 'file:', slashes: true, }) mainWindow.loadURL(startUrl) mainWindow.webContents.openDevTools({ mode: 'bottom', activate: true }) // open Web URLs in the default system browser mainWindow.webContents.on('new-window', (ev, url) => { ev.preventDefault() shell.openExternal(url) }) } async function loadAccount() { if (globalAccountID !== null) { return { id: globalAccountID, name: globalAccountName } } // Kickstart await peer.db.loaded() const id = await p(peer.db.account.findOrCreate)({ subdomain: 'person' }) globalAccountID = id await p(peer.set.load)(id) await p(peer.dict.load)(id) // Read profile const profile = peer.dict.read(id, 'profile') const name = profile?.name ?? '' globalAccountName = name peer.net.start() return { id, name } } async function setProfileName(ev, name) { await p(peer.dict.update)('profile', { name }) return name } async function writeElements(ev, elements) { if (globalAccountID === null) throw new Error('account not loaded') for (const element of elements) { await p(peer.db.feed.publish)({ account: globalAccountID, domain: 'zooboardElements', data: element, }) } } async function createInvite() { if (globalAccountID === null) throw new Error('account not loaded') let { url } = await p(peer.invite.createForFriend)({ hubs: 1, id: globalAccountID, }) // if the hub is on localhost, it's probably on the default port of 3000, so let's make things a bit easier for the user if (url.indexOf('0.0.0.0') !== -1) url = url.replace("0.0.0.0", "0.0.0.0:3000") return url } function copyToClipboard(ev, text) { clipboard.writeText(text) } let hasSubscribedToReadElements = false async function subscribeToReadElements() { if (hasSubscribedToReadElements) return hasSubscribedToReadElements = true // Load initial elements and inform renderer const elementsByID = new Map() const msgIDToElemID = new Map() for await (const { id: msgID, msg } of peer.db.records()) { if (msg.data && msg.metadata.domain === 'zooboardElements') { const { id: elemID, isDeleted } = msg.data if (isDeleted) { elementsByID.delete(elemID) } else { msgIDToElemID.set(msgID, elemID) elementsByID.set(elemID, msg.data) } } } const initialElements = [...elementsByID.values()] mainWindow.webContents.send('readElements', initialElements) // Subscribe to new elements and inform renderer peer.db.onRecordAdded(({ id: msgID, msg }) => { if (msg.data && msg.metadata.domain === 'zooboardElements') { const { id: elemID, isDeleted } = msg.data if (isDeleted) { elementsByID.delete(elemID) } else { msgIDToElemID.set(msgID, elemID) elementsByID.set(elemID, msg.data) } mainWindow.webContents.send('readElements', [msg.data]) } }) // Subscribe to deleted elements and inform renderer peer.db.onRecordDeletedOrErased((msgID) => { const elemID = msgIDToElemID.get(msgID) if (!elemID) return msgIDToElemID.delete(msgID) // Is there some other msgID that supports this elemID? If so, bail out for (const [, remainingElemID] of msgIDToElemID) { if (remainingElemID === elemID) { return } } // If not, delete the element elementsByID.delete(elemID) mainWindow.webContents.send('readElements', [ { id: elemID, isDeleted: true }, ]) }) // Finally safe to kickstart replication and garbage collection setTimeout(() => { peer.conductor.start( globalAccountID, [ ['profile@dict', 'zooboardElements@newest-100', 'hubs@set'], ['profile@dict', 'zooboardElements@newest-100'], ], 64_000 ) }, 32) } let hasSubscribedToConnections = false async function subscribeToConnections() { if (hasSubscribedToConnections) return hasSubscribedToConnections = true for await (const connections of awaitable(peer.net.peers())) { mainWindow.webContents.send('connections', connections) } } async function handlePPPPPUri(ev, uri) { if (!globalAccountID) { setTimeout(handlePPPPPUri, 100, null, uri) return } if (uri.startsWith("http:") || uri.startsWith("https://")) { uri = decodeURIComponent(uri.split('/invite#')[1]) } if (!uri.startsWith('ppppp://')) return console.log('Not a ppppp invite URI', uri) const commands = peer.invite.parse(uri) for (const command of commands) { console.log('Executing command', JSON.stringify(command)) switch (command.type) { case 'join': { try { await p(peer.hubClient.addHub)(command.multiaddr) } catch (err) { console.error('Failed to properly join hub', err) } break } case 'follow': { await p(peer.set.add)('follows', command.id) break } case 'promise.follow': { const [issuerType, issuerPubkey] = command.issuer if (issuerType !== 'pubkey') { throw new Error(`Dont know how to claim a ${issuerType} promise`) } // eslint-disable-next-line no-loop-func peer.addListener('rpc:connect', function onConnect(rpc) { if (rpc.shse.pubkey === issuerPubkey) { peer.removeListener('rpc:connect', onConnect) rpc.promise.follow(command.token, globalAccountID, (err) => { if (err) return console.error('Failed to use follow promise', err) }) } }) break } default: console.log('Unknown command type', command.type) } } } if (process.defaultApp) { if (process.argv.length >= 2) { app.setAsDefaultProtocolClient('ppppp', process.execPath, [ Path.resolve(process.argv[1]), ]) } } else { app.setAsDefaultProtocolClient('ppppp') } //const hasLock = app.requestSingleInstanceLock() //if (!hasLock) { // app.quit() //} else { // app.on('second-instance', (ev, argv, cwd, extraData) => { // if (mainWindow) { // if (mainWindow.isMinimized()) mainWindow.restore() // mainWindow.focus() // if (argv.length > 1) { // handlePPPPPUri(null, argv[argv.length - 1]) // } // } // }) app.whenReady().then(() => { ipcMain.handle('loadAccount', loadAccount) ipcMain.handle('setProfileName', setProfileName) ipcMain.handle('createInvite', createInvite) ipcMain.handle('copyToClipboard', copyToClipboard) ipcMain.handle('writeElements', writeElements) ipcMain.handle('consumeInvite', handlePPPPPUri) ipcMain.handle('subscribeToReadElements', subscribeToReadElements) ipcMain.handle('subscribeToConnections', subscribeToConnections) createWindow() if (process.argv.length > 1) { handlePPPPPUri(null, process.argv[process.argv.length - 1]) } app.on('activate', function () { if (BrowserWindow.getAllWindows().length === 0) createWindow() }) app.on('window-all-closed', function () { if (process.platform !== 'darwin') app.quit() }) }) //}