Clojure Spec用于包含非关键字键的地图

时间:2017-04-17 14:50:09

标签: clojure clojure.spec

让我们看看Leiningen项目地图的真实例子:global-vars

 ;; Sets the values of global vars within Clojure. This example
 ;; disables all pre- and post-conditions and emits warnings on
 ;; reflective calls. See the Clojure documentation for the list of
 ;; valid global variables to set (and their meaningful values).
 :global-vars {*warn-on-reflection* true
               *assert* false}

它允许leiningen的用户重新定义默认值 Clojures项目范围的全局变量。

现在,如果此地图的键由关键字组成,我们将clojure.spec/keys首先指定哪些键可以作为地图的一部分,然后分别定义这些键下的预期值。但由于clojure.spec/keys默默地忽略了:req:req-un中的非关键字,并引发:opt:opt-un的异常(从alpha15开始),我们将不得不以某种方式解决这个问题。

我们可以通过

获取大多数这些全局变量的类型
(for [[sym varr] (ns-publics 'clojure.core)
      :when (re-matches #"\*.+\*" (name sym))]
  [varr (type @varr)])
  =>
[*print-namespace-maps*     java.lang.Boolean]
[*source-path*              java.lang.String]
[*command-line-args*        clojure.lang.ArraySeq]
[*read-eval*                java.lang.Boolean]
[*verbose-defrecords*       java.lang.Boolean]
[*print-level*              nil]
[*suppress-read*            nil]
[*print-length*             nil]
[*file*                     java.lang.String]
[*use-context-classloader*  java.lang.Boolean]
[*err*                      java.io.PrintWriter]
[*default-data-reader-fn*   nil]
[*allow-unresolved-vars*    java.lang.Boolean]
[*print-meta*               java.lang.Boolean]
[*compile-files*            java.lang.Boolean]
[*math-context*             nil]
[*data-readers*             clojure.lang.PersistentArrayMap]
[*clojure-version*          clojure.lang.PersistentArrayMap]
[*unchecked-math*           java.lang.Boolean]
[*out*                      java.io.PrintWriter]
[*warn-on-reflection*       nil]
[*compile-path*             java.lang.String]
[*in*                       clojure.lang.LineNumberingPushbackReader]
[*ns*                       clojure.lang.Namespace]
[*assert*                   java.lang.Boolean]
[*print-readably*           java.lang.Boolean]
[*flush-on-newline*         java.lang.Boolean]
[*agent*                    nil]
[*fn-loader*                nil]
[*compiler-options*         nil]
[*print-dup*                java.lang.Boolean]

其余的我们可以通过阅读文档来填写。但我的问题是:我们如何编写一个规范,确保如果地图包含密钥'*assert*,它只会保留布尔值?

1 个答案:

答案 0 :(得分:4)

仅供参考,目前还没有计划支持s/keys中的非关键字密钥。

有几种方法可以做到这一点(一种方法是在验证之前或在领先的构造函数中执行clojure.walk/keywordize-keys之类的操作,然后使用s/keys)。

另一条路径是将地图视为地图条目元组的集合(某些宏可以大大清除它):

(defn warn-on-reflection? [s] #(= % '*warn-on-reflection*))
(s/def ::warn-on-reflection (s/tuple warn-on-reflection? boolean?))
(defn assert? [s] #(= % '*assert*))
(s/def ::assert (s/tuple assert? boolean?))

(s/def ::global-vars
  (s/coll-of (s/or :wor ::warn-on-reflection, :assert ::assert) :kind map?))

最后s/multi-spec尝试而不是上面的s/or可能会很有趣 - 这会使这个开放的规范可以在以后添加。以某种方式,将此规范打开可能很有用(因此也接受(s/tuple any? any?) - 因为将来可能会添加新内容或未在此处列出(例如,有动态变量来自其他名称空间)。

另外,请注意这些属性规范。例如,*unchecked-math*被视为逻辑上的真值,特别是采用逻辑真的特殊值:warn-on-boxed,但也会触发额外的行为。