I think it is reasonable to add some kind of dev-time query check to Fulcro. I think the two critical paths that would not affect performance much would be at initial mount (which would get it on hot code reload too) and in set-query!
. The latter one is used mainly by the dyn routers, so it would only cause a penalty on route changes.
I’m also feeling like we should have better errors around missing :initial-state
. That seems to trip people up a lot.
It would also be possible to add macro checks to defsc
that could find stupid errors on compile. It can’t expand the get-query
calls, but it could look for obvious mistakes, like a call to get-query
that isn’t embedded as a value in a map, a map that has more than one key, and other such obvious mistakes.
the check, for that matter, could sub calls to functions that have get-query
in their name with [:subquery-of-ComponentClassName]
and run a spec check on the resulting expression.
of course such a check would have to “bail” if there were other function calls in there
and syntax quoting from user space would cause possible problems….but I think you could catch a lot of common cases
@holyjak you have any interest in working on that? I’d be glad to give you some pointers if you’re willing to do the legwork/testing
This is on Fulcro 2
I’m not actively maintaining F2. If you want to submit a patch to fix this, please branch from the fulcro-2 branch, and send a PR targeted at that branch.
You’ll need to upgrade dependencies and test it. If you can certify that it looks good, I’ll push an update to clojars.
Hello! https://github.com/holyjak/fulcro-troubleshooting has been extended (v5, latest) to check initial state (even in lambda form) for validity and to wrap user components with error boundary so that an exception during render does not blow up the whole up. Please give it a try! https://github.com/holyjak/fulcro-troubleshooting/blob/master/CHANGELOG.md#v5---2021-03-06
yes, I do!
Speaking about the dev-time query check on mount etc. would it be (s/explain ::eql/query query)
?
As explained above, I think we would also need to check during app/set-root!
b/c it is called before mount and fails it there are errors.
BTW how do we know it is dev time? goog.DEBUG?
I think the error message would be better if we just did a valid?
and indicated that the query was not valid EQL. We might use explain-data from there and try to narrow it down, but I would not use explain.
(when goog.DEBUG …)
is the proper wrapping for it, since that will cause dead code removal in advanced compile.
And yeah, set-root!
seems a good place as well.
In the macro, I was thinking of replacing any function calls whose symbol name is get-query
as I explained above, then look for any other things that are lists that start with symbols. Then, if there are NO other symbols (recursive check) in the query, then it should be ok to check the resulting EQL and throw a syntax error. That would be way more noticeable than an error in the console.
I’m considering adding a default initial state with a join on every query edge for components with constant idents…if you wanted to try your hand at that as well 😄
hm, not sure that one is possible. It’s hard to detect if the ident is constant…nvm
some kind of runtime check around initial state seems like a good idea, I’m just not sure what it is. I can’t tell you how many times someone’s error/question is purely a matter of they didn’t set up initial state for their component.
>. I can't tell you how many times someone's error/question is purely a matter of they didn't set up initial state for their component. When is that a problem? I thought it only is critical for https://book.fulcrologic.com/#_a_warning_about_ident_and_link_queries and router targets (and all their ancestors)?
Seeing https://codesunaba.netlify.app/, would it be possible to make a fork with Fulcro? It is macro-heavy but I suppose the macros do not necessarily need to use any CLJ JVM specific code. What do you think, @tony.kay ? It would make creating interactive, hands-on Fulcro tutorials much simpler 🙂
It’s actually come up a few times as a desire (it would be cool). If memory serves correctly it isn’t just Fulcro that’s the problem…the macros actually are not that hard to port. The bigger problem was that it has a spiderweb kind of effect, where all of the dependencies have to be bootstrap compatible. So, something as simple as logging (we use timbre) cause issues, because those dependencies are also not bootstrap compatible.
app.ui=> (comp/get-initial-state app.ui/Root {})
------ WARNING - :undeclared-var -----------------------------------------------
Resource: :1:25
Use of undeclared Var app.ui/Root
--------------------------------------------------------------------------------
{:friends {:list/label "Friends", :list/people [{:person/name "Sally", :person/age 32} {:person/name "Joe", :person/age 22}]}, :enemies {:list/label "Enemies", :list/people [{:person/name "Fred", :person/age 11} {:person/name "Bobby", :person/age 55}]}}
I am going through the getting started section in the fulcro book. Do you know why this happens? (defsc Root ...) is in this ns and the returned map is correct.happens to me sometimes as well. I solve it I think by reloading the ns. ClojureScript repl has its mysteries...
hehe ok thanks 🙂