From bf26c01fd32d472825b07513918b2691be5f8bb9 Mon Sep 17 00:00:00 2001 From: Thomas Goyne Date: Tue, 20 Mar 2018 16:52:52 -0700 Subject: [PATCH] Avoid leaking realms when an error occurs within an event handler --- lib/notification-worker.js | 9 ++++++++- lib/notifier.js | 17 +++++++++++++++-- 2 files changed, 23 insertions(+), 3 deletions(-) diff --git a/lib/notification-worker.js b/lib/notification-worker.js index d6c3f030..84191b3b 100644 --- a/lib/notification-worker.js +++ b/lib/notification-worker.js @@ -43,7 +43,14 @@ process.on('message', (m) => { if (impl.onchange) { const change = Realm.Sync._deserializeChangeSet(m.change); if (!change.isEmpty) { - impl.onchange(change); + try { + impl.onchange(change); + } + catch (e) { + change.close(); + process.send({change: m.change}); + throw e; + } } change.close(); } diff --git a/lib/notifier.js b/lib/notifier.js index ea62a36c..a6bc6dd4 100644 --- a/lib/notifier.js +++ b/lib/notifier.js @@ -65,11 +65,24 @@ class FunctionListener { changes.release(); return; } - const promise = Promise.resolve(this.fn(changes)); + + let promise; + try { + promise = Promise.resolve(this.fn(changes)); + } + catch (e) { + changes.release(); + throw e; + } + this.pending.push(promise); - promise.then(() => { + const release = () => { changes.release(); this.pending.splice(this.pending.indexOf(promise), 1); + }; + promise.then(release).catch(e => { + release(); + throw e; }); } };