2011-12-25 42 views
3

我正在寫一個clojure應用程序供內部使用,我也希望配置文件也在clojure中。我定義了一些宏來使配置文件更容易寫,但是當我嘗試從配置文件中評估數據時,它無法找到我的宏。這從REPL工作正常但是。例如,我使用如何在命名空間的上下文中評估clojure數據結構?

(load-string "/path/to/config") 

我得到這個錯誤:

Exception in thread "main" java.lang.RuntimeException: Unable to resolve symbol: defcmd in this context, compiling:(null:1) 
at clojure.lang.Compiler.analyze(Compiler.java:6235) 
at clojure.lang.Compiler.analyze(Compiler.java:6177) 
at clojure.lang.Compiler$InvokeExpr.parse(Compiler.java:3452) 
at clojure.lang.Compiler.analyzeSeq(Compiler.java:6411) 
at clojure.lang.Compiler.analyze(Compiler.java:6216) 
at clojure.lang.Compiler.analyze(Compiler.java:6177) 
at clojure.lang.Compiler.eval(Compiler.java:6469) 
at clojure.lang.Compiler.load(Compiler.java:6902) 
at clojure.lang.Compiler.load(Compiler.java:6872) 
at clojure.core$load_reader.invoke(core.clj:3625) 
at clojure.core$load_string.invoke(core.clj:3635) 
at serverStats.core$load_config.invoke(core.clj:67) 
at serverStats.core$_main.doInvoke(core.clj:78) 
at clojure.lang.RestFn.invoke(RestFn.java:397) 
at clojure.lang.Var.invoke(Var.java:397) 
at user$eval109.invoke(NO_SOURCE_FILE:1) 
at clojure.lang.Compiler.eval(Compiler.java:6465) 
at clojure.lang.Compiler.eval(Compiler.java:6455) 
at clojure.lang.Compiler.eval(Compiler.java:6431) 
at clojure.core$eval.invoke(core.clj:2795) 
at clojure.main$eval_opt.invoke(main.clj:296) 
at clojure.main$initialize.invoke(main.clj:315) 
at clojure.main$null_opt.invoke(main.clj:348) 
at clojure.main$main.doInvoke(main.clj:426) 
at clojure.lang.RestFn.invoke(RestFn.java:421) 
at clojure.lang.Var.invoke(Var.java:405) 
at clojure.lang.AFn.applyToHelper(AFn.java:163) 
at clojure.lang.Var.applyTo(Var.java:518) 
at clojure.main.main(main.java:37) 
Caused by: java.lang.RuntimeException: Unable to resolve symbol: defcmd in this context 
at clojure.lang.Util.runtimeException(Util.java:156) 
at clojure.lang.Compiler.resolveIn(Compiler.java:6720) 
at clojure.lang.Compiler.resolve(Compiler.java:6664) 
at clojure.lang.Compiler.analyzeSymbol(Compiler.java:6625) 
at clojure.lang.Compiler.analyze(Compiler.java:6198) 
... 28 more 

但是,在運行從REPL在我的命名空間相同的命令工作正常。

+0

我找到了相關線索,但沒有解決辦法:http://stackoverflow.com/questions/1307567/eval-not-working-on-unexpanded-宏觀報價 – Chris 2011-12-25 02:14:19

+2

我會猜測「defcmd」在路徑字符串中的某處,因爲「load-string」沒有按照您的想法做。試試'load-file'。 – 2011-12-25 08:04:03

回答

5

您可能需要一些更復雜的加載方案。我假設你想把配置放入一個專用的配置命名空間。它只包含配置。 Helper函數保存在配置名稱空間中使用的獨立名稱空間中。

(defn setup-config-space 
    [] 
    (binding [*ns* *ns*] 
    (in-ns 'config.namespace) 
    (refer-clojure) 
    (use 'config.helpers))) 

(defn load-config 
    [path] 
    (binding [*ns* *ns*] 
    (in-ns 'config.namespace) 
    (load-file path))) 

見實例的使用:

..ojure/1.4.0-alpha3% cat config/helpers.clj        
(ns config.helpers) 

(defmacro defcmd 
    [x] 
    `(defn ~x [] "Hello")) 
..ojure/1.4.0-alpha3% cat x.clj 
(defcmd foo) 
..ojure/1.4.0-alpha3% java -cp .:clojure-1.4.0-alpha3.jar clojure.main -r 
Clojure 1.4.0-alpha3 
user=> ; Paste above functions 
#'user/setup-config-space 
#'user/load-config 
user=> (setup-config-space) 
nil 
user=> (load-config "x.clj") 
#'config.namespace/foo 
user=> (config.namespace/foo) 
"Hello" 
+0

你能解釋一下,爲什麼在REPL中評估:(binding [* ns *(find-ns'config.namespace)](defn tst [] 1))#'user/tst而不是#'config.namespace/tst ? – 2011-12-25 17:41:09

+0

因爲'def'的var創建部分(來自'defn')是在編譯命令時完成的,但'binding'只發生在執行主體時。因此,名稱空間仍然是「用戶」。 ''def。*'在非頂層位置大部分時間都是錯誤的。 (例外:通過let綁定的值關閉,'(let [x ...](defn foo [] x))') – kotarak 2011-12-25 23:55:40

+0

好的提示。真正使用它。在這裏可以看到一個小例子:https://gist.github.com/paulosuzart/4727858#file-init-clj – paulosuzart 2013-02-07 02:22:09