This page is not created by, affiliated with, or supported by Slack Technologies, Inc.
2020-11-30
Channels
- # adventofcode (4)
- # aleph (1)
- # announcements (7)
- # aws (10)
- # babashka (23)
- # beginners (23)
- # calva (20)
- # chlorine-clover (13)
- # cider (17)
- # clj-kondo (13)
- # cljfx (9)
- # cljsrn (9)
- # clojure (98)
- # clojure-australia (1)
- # clojure-dev (15)
- # clojure-europe (127)
- # clojure-nl (4)
- # clojure-sanfrancisco (3)
- # clojure-uk (98)
- # clojurescript (25)
- # community-development (8)
- # core-async (15)
- # cryogen (9)
- # cursive (7)
- # data-science (1)
- # datascript (5)
- # datomic (3)
- # devcards (2)
- # fulcro (5)
- # graalvm (1)
- # helix (8)
- # jackdaw (1)
- # jobs (5)
- # kaocha (17)
- # malli (5)
- # meander (5)
- # off-topic (37)
- # pathom (33)
- # pedestal (3)
- # re-frame (12)
- # reitit (1)
- # remote-jobs (3)
- # sci (1)
- # shadow-cljs (6)
- # testing (1)
- # vim (6)
- # vrac (5)
I'm chatting with someone who is trying to use Java interop on a class where there is 1 method in a class with a particular name and 1 parameter, and it extends an abstract class that has several more methods with the same name and 1 parameter (different types for that parameter). When he is trying to call a method on an instance of the child class, but the method he wants to call is in the abstract parent class, Clojure reflection seems to prefer non-bridge methods over bridge methods, and all of the methods in the parent class are bridge methods in the child class. Even with an explicit type hint on the 1 parameter in the Clojure interop call, it gives a ClassCastException at Java interop call time, because it cannot cast the type of the parameter in the interop call to the type of a different method than the one the developer is trying to call. More details in thread.
Here is a link to the discussion (I hope, if I am doing this correctly):
Public git repo of @U3FKR3KDF code: https://github.com/stuartrexking/pdfboxing
Minimal repro is to fire up a REPL using the deps.edn file in the root dir of that repo, then evaluate these two expressions:
(require '[ventures.fierce.pdfbox.core :as c])
(in-ns 'ventures.fierce.pdfbox.core)
then eval the expression in the comment block at the end of the core.clj source file.
OK, maybe not minimal repro, but at least very quick to get to the exception.
The method named setNonStrokingColor
has 1 signature in the class PDAppearanceContentStream
here: https://github.com/apache/pdfbox/blob/trunk/pdfbox/src/main/java/org/apache/pdfbox/pdmodel/PDAppearanceContentStream.java#L168
and that one is the one that the Clojure Reflector code seems to find, even though it has the wrong type. The others are in the abstract parent class here: https://github.com/apache/pdfbox/blob/fffad2fe3ff74eaca909fce65cf1dd4191ceb036/pdfbox/src/main/java/org/apache/pdfbox/pdmodel/PDAbstractContentStream.java#L76
I am not sure if this is a case that Clojure reflection is intended to handle correctly, or not.
reminds me of an old issue with reflection and bridge methods, let me see if I can find it.
https://clojure.atlassian.net/browse/CLJ-1243 is the one I'm thinking of
https://ask.clojure.org/index.php/4255/cannot-resolve-public-generic-method-package-private-class is the equivalent ask clojure page for public voting or comments
Thanks for the links, Alex! The description makes it sound like it could very well be exactly that issue. The workaround mentioned by one commenter is one that I mentioned to @U3FKR3KDF - define a small class in Java that calls only the java methods that you need. Make everything you need public, and then if it works, use Java interop to call those methods. A pain, but probably effective.
Thanks for that. I’ve created a class to manage the interop.