The way I use mixins is really simple and might be a bit too specific to how I currently use channels I plan to explore the idea more, but haven't had the time yet. I'm sure you can build much more sophisticated abstractions.
I don't have the code in front of me right now, but the basic idea is as follows: Before om-tools, I had a subscribe function that wrapped core.async/sub. You pass it a keyword topic, owner and a function that takes the value as its argument. All messages on the channel take the form [topic value]. I also had a matching unsubscribe function that took owner and killed the go blocks created by all subscribe calls in that component. The problem I had was that it was very easy to forgot to call unsubscribe in IWillUnmount (plus the extra boilerplate was a bit annoying). So I wrote a mixin that looks something like this: (defmixin subscriber (will-unmount [owner] (my-unsubscribe owner)) (subscribe [owner topic callback] (my-subscribe owner topic callback))) Which can then be used like this: (defcomponentk a-component [owner] (:mixins subscriber) (did-mount [_] (.subscribe owner :foo (fn [v] ...))) (render [_] ...)) For your purposes, perhaps something like this would work: (defmixin go-block-aware (init-state [] {:exit-chan (async/chan)}) (will-unmount [owner] (async/put! (om/get-state owner :exit-chan) true)) (go [owner callback] (let [exit-chan (om/get-state owner :exit-chan)] (go-loop [] (let [[v ch] (async/alts! [dump-chan exit-chan])] (when (= ch dump-chan) (callback v) (recur))))))) And use like this: (defcomponentk a-component [owner] (:mixins go-block-aware) (did-mount [_] (.go owner (fn [v] ...))) (render [_] ...)) On 25 July 2014 19:12, Alexander K. Hudek <alexan...@hudek.org> wrote: > Could you describe your mixin a bit more? We’ve just written an om > component and macro to help clean up go-blocks in cases like these. Several > of our components take channels as input from parent components to allow > for more sophisticated communication. We’ll post a link to this manager > component today or tomorrow after we add some documentation. > > Regarding the second problem, we don’t typically do that either. I was > just being lazy when writing the error demo and ran into. Curious > behaviour, but easy to avoid the problem. > > > From: Daniel Kersten <dkers...@gmail.com> <dkers...@gmail.com> > Reply: clojure@googlegroups.com <clojure@googlegroups.com>> > <clojure@googlegroups.com> > Date: July 25, 2014 at 5:26:56 AM > To: clojure@googlegroups.com <clojure@googlegroups.com>> > <clojure@googlegroups.com> > Subject: Re: subtle om + core.async problems > > You could simplify your fix code a small bit by using go-loop and when, > like this: > > (go-loop [] > (let [[v ch] (alts! [dump-chan (om/get-state owner :exit-chan)])] > (when (= ch dump-chan) > (.log js/console "dumping state:") > (.log js/console (pr-str (om/get-state owner))) > (recur)))) > > This won't work in your code because my-form's parent owns dump-chan, but > if a component owns the channel, an alternative approach is to simply close > dump-chan in IWillUnmount and change your go block to something like this: > > (go-loop [] > (when-let [v (<! dump-chan)] > (.log js/console "dumping state:") > (.log js/console (pr-str (om/get-state owner))) > (recur))) > > In my own code, I've abstracted my go blocks into om-tools mixins that > handle the killing of go blocks. > > I'm not sure why the second problem happens. As a general rule, I never > create new resources in render (IMO render should be pure functional: app > state and local state in -> new dom nodes out; but as the child stores the > channel between calls to parents render, this breaks that "rule") and I've > never had the issue. > Since base isn't being re-rendered in this case, I'm not sure why this > would be a problem here. > > > > On 24 July 2014 02:35, Sean Corfield <s...@corfield.org> wrote: > >> You'll want to read this thread: >> https://groups.google.com/forum/#!topic/clojurescript/DHJvcGey8Sc >> >> In particular: >> >> "So if you have code that's like this, those components will want to >> clean up after themselves in IWillUnmount." >> >> That should address your first problem? >> >> I'm not sure what to suggest right now about the second problem. >> >> Sean >> >> >> On Jul 23, 2014, at 4:19 PM, Alexander Hudek <alexan...@hudek.org> wrote: >> >> I've encountered two subtle but serious problems using om with >> core.async. >> >> The first one is illustrated by this code: >> >> https://github.com/akhudek/om-async-error >> >> First, one obvious solution here is to move the dump-chan inside the form >> state. >> However, it's written this way to illustrate the error which originally >> arose in a >> dialog component that took both an action channel that receives button >> presses >> and a dialog content component. It exposed the action channel in this way >> in >> order to be flexible. >> >> I believe the cause of this error is that when you toggle the form in the >> demo >> to off, it unmounts the component. However, the go-block is still active >> and >> listening to the channel. When you toggle the form back on, a new >> component >> is created and mounted. Now you have two components listening on the same >> channel! >> >> The ideal solution might be to find a way to end the go block when the >> component >> unmounts. This is easy to do on a case by case basis, but not easy to do >> in a >> completely generic fashion. Here is one solution: >> >> >> https://github.com/akhudek/om-async-error/blob/alts-fix/src/om_async_error/core.cljs >> >> The second problem is easy to fix, however, I don't understand why it >> happens. >> >> https://github.com/akhudek/om-async-error/tree/chan-opts-bug >> >> Instead of initializing the dump channel in the base components state, we >> initialize >> it when creating the form component. As soon as you start typing into the >> form >> the channel breaks. It was suggested on irc that this might be because of >> a >> re-render of base would create a new channel, but the form component will >> not >> be re-mounted because it's already mounted. However, putting in some debug >> code shows that there is no re-render of the base component occurring >> between >> when the channel works and when it doesn't. >> >> I'm posting this here mostly to warn people to be wary of these >> situations. Of >> course, if you have any suggestions or explanations for these I'd love to >> hear them. >> >> Alex >> >> >> >> > -- > You received this message because you are subscribed to the Google > Groups "Clojure" group. > To post to this group, send email to clojure@googlegroups.com > Note that posts from new members are moderated - please be patient with > your first post. > To unsubscribe from this group, send email to > clojure+unsubscr...@googlegroups.com > For more options, visit this group at > http://groups.google.com/group/clojure?hl=en > --- > You received this message because you are subscribed to a topic in the > Google Groups "Clojure" group. > To unsubscribe from this topic, visit > https://groups.google.com/d/topic/clojure/abczlIGvogk/unsubscribe. > To unsubscribe from this group and all its topics, send an email to > clojure+unsubscr...@googlegroups.com. > > For more options, visit https://groups.google.com/d/optout. > > -- > You received this message because you are subscribed to the Google > Groups "Clojure" group. > To post to this group, send email to clojure@googlegroups.com > Note that posts from new members are moderated - please be patient with > your first post. > To unsubscribe from this group, send email to > clojure+unsubscr...@googlegroups.com > For more options, visit this group at > http://groups.google.com/group/clojure?hl=en > --- > You received this message because you are subscribed to the Google Groups > "Clojure" group. > To unsubscribe from this group and stop receiving emails from it, send an > email to clojure+unsubscr...@googlegroups.com. > For more options, visit https://groups.google.com/d/optout. > -- You received this message because you are subscribed to the Google Groups "Clojure" group. To post to this group, send email to clojure@googlegroups.com Note that posts from new members are moderated - please be patient with your first post. To unsubscribe from this group, send email to clojure+unsubscr...@googlegroups.com For more options, visit this group at http://groups.google.com/group/clojure?hl=en --- You received this message because you are subscribed to the Google Groups "Clojure" group. To unsubscribe from this group and stop receiving emails from it, send an email to clojure+unsubscr...@googlegroups.com. For more options, visit https://groups.google.com/d/optout.