2024 年 Clojure 状态调查! 中分享您的想法。

欢迎!请参阅 关于 页面以获取此工作方式的更多信息。

0
错误
当 *compile-path* 未设置为 String,而是设置为 java.io.File(这正是 boot-clj 目前所做的事情)。当在 boot repl 中尝试使用时,编译器会抛出 Cast 异常。常识似乎是要避免从 repl 中编译,人们说不要使用 gen-class。我发现当您想快速迭代并使用 gen-class 时,这很不幸。

我没有测试所有较老的版本是否存在此问题,但 git blame 显示在此处有帮助的代码过去 10 年没有更改。
https://github.com/clojure/clojure/blame/master/src/clj/clojure/core.clj#L6073
https://github.com/clojure/clojure/blame/master/src/jvm/clojure/lang/Compiler.java#L7643

我现在的工作方式是


(defn compile-safely [ns-name]
  (let [compile-path-class *compile-path*
        corrected-path (cond
                         (instance? java.io.File compile-path-class) (.getAbsolutePath *compile-path*)
                         (instance? String compile-path-class)      *compile-path*
                         :else
                         (throw (ex-info "不支持的类" {:class (class compile-path-class)})))
    (binding [*compile-path* corrected-path]
      (compile ns-name))))


我建议在 'clojure.core/compile 函数中添加类型检查。

5 答案

0
0

由 jeroen 制作的评论

以下在boot-clj中有效(由于类路径问题,不在Leiningen和clj cli中): https://gist.github.com/jeroenvandijk/8187413d24433545eeb9579538a903f7#file-repl_compile-clj-L39-L52

0

评论者:alexmiller

您能提供一个可重现的示例吗?

看起来ClassCastException对于拥有错误类型的类是非常准确的。

0
评论者:jeroen

我已经尽量让示例尽可能简单。您说得对,ClassCastException是非常准确的,但您只有在弄清楚它与*compile-path*有关之后才会意识到这一点。这个错误报告主要关于管理(Clojure用户的)期望。



clj -A:new app example.error
cd example.error
clj

 ;; clj已经将*compile-path*正确设置为字符串(尽管不在类路径中)
 *compile-path* ;=> "classes"
 (let [f (clojure.java.io/file "classes/foo.clj")]
   (clojure.java.io/make-parents f)
   (spit f "(ns foo) (gen-class :name my.new.Class)")
   (compile 'foo)
   (clojure.java.shell/sh "ls" "classes"))
 ; 成功,尽管您仍然无法加载该类
 ; #=> {:exit 0, :out  "clojure\nfoo$fn__134.class\nfoo$fn__166.class\nfoo$fn__174.class\nfoo$loading__6549__auto____132.class\nfoo$loading__6549__auto____164.class\nfoo$loading__6549__auto____172.class\nfoo.clj\nfoo__init.class\n", :err ""}
 

 ;; 从一个boot项目
 ;; boot.user=> *compile-path*
 ;; #object[java.io.File 0x5266b1a "/var/folders/ck/9zqvs0zx5vsf2bc7rvv1pwc40000gn/T/boot-repl8170014881711480931"]
 ;; 这可以在clj repl中重现
 (binding [*compile-path* (clojure.java.io/file "classes")]
   (compile 'foo)
   )

 编译fn*时发生语法错误(在foo.clj:1:1)。
 原因:无法将java.io.File转换为java.lang.String
 
 
0
参考: https://clojure.atlassian.net/browse/CLJ-2412(由jeroen报告)
...