2011-12-25 41 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