使用lein uberjar生成的jar在NoClassDefFoundError上失败

时间:2010-12-04 15:37:44

标签: clojure compojure leiningen

我有一个简单的web应用程序与这个project.clj:

(defproject squirrel-money "1.0.0-SNAPSHOT"
  :description "Squirrel Money"
  :dependencies [[org.clojure/clojure "1.2.0"]
                 [org.clojure/clojure-contrib "1.2.0"]
                 [compojure "0.5.3"]
                 [ring/ring-jetty-adapter "0.3.5"]
                 [hiccup "0.3.1"]
                 [postgresql "8.4-701.jdbc4"]
                 [clj-time "0.2.0-SNAPSHOT"]]
  :dev-dependencies [[lein-eclipse "1.0.0"]]
  :main squirrel-money.main
  :repl-init-script "src/squirrel_money/init_repl.clj")

我的主要看起来像这样:

(ns squirrel-money.main
  (:gen-class)
  (:use 
    [compojure.core]
    [ring.adapter.jetty])
  (:require 
    [compojure.route :as route]
    [squirrel-money.savings :as savings]))

(defn launch [routedef]
  (run-jetty routedef {:port 17080}))

(defroutes money-routes
  (GET "/savings" [] (savings/render))
  (route/not-found "Page not found"))

(defn -main [& args] (launch money-routes))

使用REPL工作正常。但是,当我生成一个带有lein uberjar的jar并尝试执行它时:

java -jar squirrel-money-1.0.0-SNAPSHOT-standalone.jar

它死于这个例外:

Exception in thread "main" java.lang.NoClassDefFoundError: compojure/response/Renderable
    at squirrel_money.main$fn__1067.invoke(main.clj:18)
    at squirrel_money.main__init.load(Unknown Source)
    at squirrel_money.main__init.<clinit>(Unknown Source)
    at java.lang.Class.forName0(Native Method)
    at java.lang.Class.forName(Class.java:247)
    at clojure.lang.RT.loadClassForName(RT.java:1578)
    at clojure.lang.RT.load(RT.java:399)
    at clojure.lang.RT.load(RT.java:381)
    at clojure.core$load$fn__4511.invoke(core.clj:4905)
    at clojure.core$load.doInvoke(core.clj:4904)
    at clojure.lang.RestFn.invoke(RestFn.java:409)
    at clojure.lang.Var.invoke(Var.java:365)
    at squirrel_money.main.<clinit>(Unknown Source)
Caused by: java.lang.ClassNotFoundException: compojure.response.Renderable
    at java.net.URLClassLoader$1.run(URLClassLoader.java:202)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:190)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:307)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:301)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:248)
    ... 13 more

我做错了什么?如何让它工作?

不确定这是否重要,但我注意到在jar中我的文件,clojure本身和Java库被解压缩为.class文件,而所有clojure库仅作为普通的.clj文件存在。

1 个答案:

答案 0 :(得分:2)

这似乎是一个leinigen 1.4.0错误。您可能想尝试使用leiningen 1.3.1创建一个uberjar。

修改

Leiningen 1.4.0删除非项目.class文件以解决Clojure错误(请参阅CLJ-322)。显然,这种行为有时会导致问题。

您可以通过在.class中将:keep-non-project-classes设置为true来保持leiningen 1.4.0不删除非项目project.clj文件。

有关详细信息,请参阅相关的leinigen issue