什么是Clojure的生成测试?

Ert*_*tin 6 functional-programming clojure clojurescript generative-testing clojure.spec

我偶然发现了这个Generative Testing in Clojure with spec想法,并希望了解它.

提供一些例子也非常有用.

Rov*_*ion 15

作为介绍性阅读,我们已经获得了理论和概述以及指南,它应该为您提供有关原因和方式的信息.

如果你想要一个有点复杂的例子,我们可以采取以下string->semantic-version功能leiningen.release:

(defn string->semantic-version [version-string]
  "Create map representing the given version string. Returns nil if the
  string does not follow guidelines setforth by Semantic Versioning 2.0.0,
  http://semver.org/"
  ;; <MajorVersion>.<MinorVersion>.<PatchVersion>[-<Qualifier>][-SNAPSHOT]
  (if-let [[_ major minor patch qualifier snapshot]
           (re-matches
            #"(\d+)\.(\d+)\.(\d+)(?:-(?!SNAPSHOT)([^\-]+))?(?:-(SNAPSHOT))?"
            version-string)]
    (->> [major minor patch]
         (map #(Integer/parseInt %))
         (zipmap [:major :minor :patch])
         (merge {:qualifier qualifier
                 :snapshot snapshot}))))
Run Code Online (Sandbox Code Playgroud)

它需要一个字符串并尝试将其解析为一个程序可读的映射,表示某个工件的版本号.它的规格可能如下:

首先是一些依赖

(ns leiningen.core.spec.util
  (:require
   [clojure.spec           :as spec]
   [clojure.spec.gen       :as gen]
   [miner.strgen           :as strgen]
   [clojure.spec.test      :as test]
   [leiningen.release      :as release]))
Run Code Online (Sandbox Code Playgroud)

然后是一个帮助宏

(defmacro stregex
  "Defines a spec which matches a string based on a given string
  regular expression. This the classical type of regex as in the
  clojure regex literal #\"\""
  [string-regex]
  `(spec/with-gen
     (spec/and string? #(re-matches ~string-regex %))
     #(strgen/string-generator ~string-regex)))
Run Code Online (Sandbox Code Playgroud)

然后是语义版本的定义

(spec/def ::semantic-version-string
  (stregex #"(\d+)\.(\d+)\.(\d+)(-\w+)?(-SNAPSHOT)?"))
Run Code Online (Sandbox Code Playgroud)

和一些助手规格

(spec/def ::non-blank-string
  (spec/and string? #(not (str/blank? %))))
(spec/def ::natural-number
  (spec/int-in 0 Integer/MAX_VALUE))
Run Code Online (Sandbox Code Playgroud)

用于在结果映射中定义键

(spec/def ::release/major     ::natural-number)
(spec/def ::release/minor     ::natural-number)
(spec/def ::release/patch     ::natural-number)
(spec/def ::release/qualifier ::non-blank-string)
(spec/def ::release/snapshot  #{"SNAPSHOT"})
Run Code Online (Sandbox Code Playgroud)

和地图本身

(spec/def ::release/semantic-version-map
  (spec/keys :req-un [::release/major ::release/minor ::release/patch
                      ::release/qualifier ::release/snapshot]))
Run Code Online (Sandbox Code Playgroud)

其次是功能规范:

(spec/fdef release/string->semantic-version
           :args (spec/cat :version-str ::release/semantic-version-string)
           :ret  ::release/semantic-version-map)
Run Code Online (Sandbox Code Playgroud)

到目前为止,我们可以让Clojure Spec生成测试数据并将其提供给函数本身,以测试它是否满足我们为它设置的约束:

(test/check `release/version-map->string)
=> ({:spec #object[clojure.spec$fspec_impl$reify__14248 0x16c2555 "clojure.spec$fspec_impl$reify__14248@16c2555"],
     :clojure.spec.test.check/ret {:result true,
                                   :num-tests 1000,
                                   :seed 1491922864713},
     :sym leiningen.release/version-map->string})
Run Code Online (Sandbox Code Playgroud)

这告诉我们,在为我们生成的1000个测试用例规范中,函数通过了每一个.


Ala*_*son 6

clojure/test.check在潜入之前,您可能会发现最容易开始查看Clojure Spec. 从项目页面:

(require '[clojure.test.check :as tc])
(require '[clojure.test.check.generators :as gen])
(require '[clojure.test.check.properties :as prop])

(def sort-idempotent-prop
  (prop/for-all [v (gen/vector gen/int)]
    (= (sort v) (sort (sort v)))))

(tc/quick-check 100 sort-idempotent-prop)
;; => {:result true, :num-tests 100, :seed 1382488326530}
Run Code Online (Sandbox Code Playgroud)

在散文中,该测试读取:对于所有整数向量,v,排序v等于排序v两次.

如果我们的测试失败会怎么样?test.check将尝试找到仍然失败的"较小"输入.这个过程称为收缩.让我们看看它的实际效果:

(def prop-sorted-first-less-than-last
  (prop/for-all [v (gen/not-empty (gen/vector gen/int))]
    (let [s (sort v)]
      (< (first s) (last s)))))

(tc/quick-check 100 prop-sorted-first-less-than-last)
;; => {:result false, :failing-size 0, :num-tests 1, :fail [[3]],
       :shrunk {:total-nodes-visited 5, :depth 2, :result false,
                :smallest [[0]]}}
Run Code Online (Sandbox Code Playgroud)