This page is not created by, affiliated with, or supported by Slack Technologies, Inc.
- # announcements (15)
- # bangalore-clj (1)
- # beginners (207)
- # calva (22)
- # cider (4)
- # clara (73)
- # cljs-dev (7)
- # cljsrn (4)
- # clojure (125)
- # clojure-dev (38)
- # clojure-europe (2)
- # clojure-india (11)
- # clojure-italy (11)
- # clojure-nl (14)
- # clojure-russia (22)
- # clojure-uk (32)
- # clojurescript (30)
- # cursive (11)
- # datavis (2)
- # datomic (14)
- # editors (3)
- # emacs (3)
- # hyperfiddle (4)
- # juxt (13)
- # klipse (1)
- # luminus (5)
- # nrepl (7)
- # off-topic (9)
- # overtone (13)
- # portkey (1)
- # re-frame (15)
- # reagent (13)
- # ring (30)
- # schema (4)
- # shadow-cljs (108)
- # spacemacs (8)
- # specter (3)
- # sql (2)
- # testing (11)
- # tools-deps (21)
- # unrepl (4)
Does clara supports creation of a session and firing rules in different threads? I'm having an exception trying to do it:
Exception in thread "async-dispatch-24" java.lang.IllegalArgumentException: find not supported on type: clara.rules.engine.Token$reify__26254
@mauricio.szabo I’d expect it to be ok. Can I see (1) more stack trace and (2) an (perhaps simplified) example of how you are setting up for this?
Are you creating more than one session at a time? The compiler uses eval calls internally, and the Clojure compiler itself isn’t thread-safe.
@ I've tried creating more than one, and don't creating it, and they both gave me the same error
@ I'm running the code in the following way:
(let [session (-> (mk-session 'events-reports.reports.deferred 'events-reports.reports.normalize-data :fact-type-fn :fact)) res (jdbc/query db "SELECT * FROM my_table WHERE ...") location-ids (map :id res)] (doseq [id location-ids] (async/go (println "Emitting for" id) (jdbc/with-db-connection [db @smart/db] (let [facts (search-facts db id) rows (-> session (insert-all facts) (fire-rules) (query detailed-anual-report) (->> (map make-detailed-anual-row)))] (println "Saving for" id) (save-rows rows))))))
Exception in thread "async-dispatch-4" java.lang.IllegalArgumentException: find not supported on type: clara.rules.engine.Token$reify__26254 at clojure.lang.RT.find(RT.java:863) at clojure.core$select_keys.invokeStatic(core.clj:1539) at clojure.core$select_keys.invoke(core.clj:1532) at clara.rules.engine$propagate_items_to_nodes$fn__26782.invoke(engine.cljc:154) at clara.rules.platform$group_by_seq$fn__26030.invoke(platform.cljc:39) at clojure.core.protocols$naive_seq_reduce.invokeStatic(protocols.clj:62) at clojure.core.protocols$interface_or_naive_reduce.invokeStatic(protocols.clj:72) at clojure.core.protocols$fn__7852.invokeStatic(protocols.clj:169) at clojure.core.protocols$fn__7852.invoke(protocols.clj:124) at clojure.core.protocols$fn__7807$G__7802__7816.invoke(protocols.clj:19) at clojure.core.protocols$seq_reduce.invokeStatic(protocols.clj:31) at clojure.core.protocols$fn__7835.invokeStatic(protocols.clj:75) at clojure.core.protocols$fn__7835.invoke(protocols.clj:75) at clojure.core.protocols$fn__7781$G__7776__7794.invoke(protocols.clj:13) at clojure.core$reduce.invokeStatic(core.clj:6748) at clojure.core$reduce.invoke(core.clj:6730) at clara.rules.platform$group_by_seq.invokeStatic(platform.cljc:38) at clara.rules.platform$group_by_seq.invoke(platform.cljc:27) at clara.rules.engine$propagate_items_to_nodes.invokeStatic(engine.cljc:154) at clara.rules.engine$propagate_items_to_nodes.invoke(engine.cljc:147) at clara.rules.engine.LocalTransport.send_elements(engine.cljc:176) at clara.rules.engine.AlphaNode.alpha_activate(engine.cljc:538) at clara.rules.engine$flush_updates$flush_all__26940.invoke(engine.cljc:269) at clara.rules.engine$flush_updates.invokeStatic(engine.cljc:277) at clara.rules.engine$flush_updates.invoke(engine.cljc:252) at clara.rules.engine$fire_rules_STAR_.invokeStatic(engine.cljc:1825) at clara.rules.engine$fire_rules_STAR_.invoke(engine.cljc:1720) at clara.rules.engine.LocalSession.fire_rules(engine.cljc:1909) at clara.rules$fire_rules.invokeStatic(rules.cljc:44) at clara.rules$fire_rules.invoke(rules.cljc:29)
actually, must be thinking about that wrong, since you can’t
clara.rules.engine.Token, it’s a record
that is just relating to a reify happening from within the clj record impl I see - disregard those comments then,
I think I’ve recreated it to some minimal degree as far as what could cause that type of exception
I believe the
getLookupThunk can happen via clj keyword lookup compile-time inlining stuff
so this ends up coming from a call
(:bindings <token-obj>) in the fn
Clojure version: Clojure 1.9.0 JVM version: Eclipse OpenJ9 VM 1.8.0_181-b13 Core.Async version: 0.4.490
this problem is weird. asked a bit on #clojure-dev to see if there were any thoughts on how you could end up with a keyword callsite messing up like this. The only possible thing that comes to mind so far is just making sure that you aren’t somehow requiring clara.rules.engine (or other ns’s) multiple times
so somehow getting different versions of the same classes loaded into successive clj dynamic class loaders
It sort of looks to me now that you may have
clojure.rules.engine being loaded more than once, and causing a duplicate class of clojure.rules.engine.Token being created.
I think I have seen this behavior before, but i need to verify my suspicions before I could say for sure.
If it is what i'm thinking, its a scenario where facts are inserted but have no corresponding nodes to activate. Thus meaning all work in prior rules was "unnecessary" and can be retracted. It sounds weird making this comment, but i think it relates to keeping the memory as low as possible... Again, i need to verify this and i am writing a simple set of rules to test it
so using logical insert (the default) can/will lead to them as the TMS keeps the working state’s “truth” consistent
Seeing your screenshot there though - relative to the other times I see - it looks like a lot of time is being spent in equality comparisons.
However, the memory updating functions comparing tokens and “remove first of each” do not seem to be as high.
I’d have to expand on the callstack trees of these a bit more perhaps to really know what’s going on there though
@ we made a mistake in a parameter in the call we are timing. After fixing, we don't see a high alpha-retract time.
@ what are you using to profile Clara? I've tried to use VisualVM but my code hang even with small group of facts...
I don’t try to use the more invasive profiler there - I don’t even know that I’d call that “recommended”
@ the example you showed did not look to have a high time spent in alpha-retract, so confused why you considered it a problem there
or maybe I was reading this wrong. I tend to look at the callstack based view - so nvm I think
yep, just noticed. Harder to read that to me. but Yeah, that means it is the “entry point” to a lot of the stuff taking the higher self-times below
To my original comment, the scenario i was remembering was something like:
This does cause an retract on the production node, but thats due to new facts being add. So not as i remembered.
(defrule simple-rule [A (= ?f f)] [?bs <- (acc/all) :from [B (= f ?f)]] => (insert! (->C)))
@ It’s a bit cutoff and always a challenge to really know.
From this, it seems to suggest that you have a rule with a negation -
:not that is initially true due to no matches at some point in fire-rules. Later, it gets matches so becomes false.
Due to order of rule firing, when it is becomes false, there was already staged activations of the rule from when it was true that need to be discarded now that that is no longer the case - via the truth maintenance
The activations being discarded may have large tokens as one possibility - perhaps there is a large data gathered via an accumulator in this rule?
Alternatively, I may have to see more stack to know (or can’t really know), the rule had a lot of combinations of matches resulting in many activations that now all have to be discarded
Maybe you can find one of that structure. The negation May be your actual last condition of the role, although that isn’t strictly true in all cases I believe.
May be able to do some listener tracing to find more details too. I’d have to play around to come up with a useful example of how to use it in this sort of case.
Regarding tracing - I wrote some helpers at one point for this that I really need to clean up some and push to master - but basically you get an entry in a list for every interaction in the rules engine, most of which have a node ID attached. You can then use that ID to look up the node in the session and see what rules/queries it corresponds to. So if you see an ID come up a lot those rules are doing a lot of work. Since it is ordered, you can investigate in more complex ways, but often a “lots of stuff is happening in this rule” can be enough to narrow things down enough