项目作者: amperity

项目描述 :
Clojure integration testing framework
高级语言: Clojure
项目地址: git://github.com/amperity/greenlight.git
创建时间: 2018-03-12T23:15:28Z
项目社区:https://github.com/amperity/greenlight

开源协议:Other

下载


Greenlight

CircleCI

This library provides an integration testing framework for Clojure. Running
a suite of tests against your systems gives you the confidence to greenlight
them for promotion to production. The primary goals of this framework are:

  • Steps should be composable to drive down repetition in similar tests.
  • Tests should support parallelization out-of-the-box.
  • Results should be actionable and easy to understand.

Installation

Library releases are published on Clojars. To use the latest version with
Leiningen, add the following to your project dependencies:

Clojars Project

Usage

A quick overview of greenlight usage:

  1. (require
  2. '[greenlight.test :as test :refer [deftest]]
  3. '[greenlight.step :as step :refer [defstep]]
  4. '[greenlight.runner :as runner]
  5. '[clojure.test :refer [is]])
  6. ;; Greenlight tests are built from a collection of steps
  7. (defstep math-test
  8. "A simple test step"
  9. :title "Simple Math Test"
  10. :test (fn [_] (is (= 3 (+ 1 2)))))
  11. => #'user/math-test
  12. (deftest simple-test
  13. "A simple test of addition"
  14. (math-test))
  15. => #'user/simple-test
  16. ;; Tests and steps are just data
  17. (simple-test)
  18. => {:greenlight.test/description "A simple test of addition",
  19. :greenlight.test/line 14,
  20. :greenlight.test/ns user,
  21. :greenlight.test/steps [{:greenlight.step/inputs {},
  22. :greenlight.step/name math-test,
  23. :greenlight.step/test #<Fn@7bb15aaa user/math_test[fn]>,
  24. :greenlight.step/title "Simple Math Test"}],
  25. :greenlight.test/title "simple-test"}
  26. ;; Tests can be ran individually
  27. (test/run-test! {} (simple-test))
  28. => {:greenlight.test/context {},
  29. :greenlight.test/description "A simple test of addition",
  30. :greenlight.test/ended-at #<java.time.Instant@55e7469c 2018-07-01T17:03:29.811Z>,
  31. :greenlight.test/line 14,
  32. :greenlight.test/ns user,
  33. :greenlight.test/outcome :pass,
  34. :greenlight.test/started-at #<java.time.Instant@224450d6 2018-07-01T17:03:29.808Z>,
  35. :greenlight.test/steps [{:greenlight.step/cleanup [],
  36. :greenlight.step/elapsed 0.002573744,
  37. :greenlight.step/inputs {},
  38. :greenlight.step/message "1 assertions (1 pass)",
  39. :greenlight.step/name math-test,
  40. :greenlight.step/outcome :pass,
  41. :greenlight.step/reports [{:actual (3),
  42. :expected 3,
  43. :message nil,
  44. :type :pass}],
  45. :greenlight.step/test #<Fn@2be25eaa user/math_test[fn]>,
  46. :greenlight.step/title "Simple Math Test"}],
  47. :greenlight.test/title "simple-test"}
  48. ;; Or as part of a suite with configurable reporters
  49. (runner/run-tests! (constantly {}) [(simple-test)] {})
  50. ; Starting test system...
  51. ; Running 1 tests...
  52. ;
  53. ; * Testing simple-test
  54. ; | user:124
  55. ; | A simple test of addition
  56. ; |
  57. ; +->> Simple Math Test
  58. ; | 1 assertions (1 pass)
  59. ; | [PASS] (0.000 seconds)
  60. ; |
  61. ; |
  62. ; * PASS (0.001 seconds)
  63. ;
  64. ;
  65. ; Ran 1 tests containing 1 steps with 1 assertions:
  66. ; * 1 pass
  67. => true

Test System

Tests are executed with a system
of components. The component lifecycle is managed by the test runner: components
are started on test startup and then stopped on test completion.

  1. (require '[com.stuartsierra.component :as component])
  2. (defrecord TestComponent
  3. []
  4. component/Lifecycle
  5. (start [this]
  6. (println "Starting test component")
  7. this)
  8. (stop [this]
  9. (println "Stopping test component")
  10. this))
  11. (runner/run-tests!
  12. (constantly (component/system-map ::component (->TestComponent)))
  13. [(simple-test)]
  14. {})
  15. ; Starting test system...
  16. ; Starting test component
  17. ; Running 1 tests...
  18. ;
  19. ; * Testing simple-test
  20. ; | user:690
  21. ; | A simple test of addition
  22. ; |
  23. ; +->> Simple Math Test
  24. ; | 1 assertions (1 pass)
  25. ; | [PASS] (0.000 seconds)
  26. ; |
  27. ; |
  28. ; * PASS (0.001 seconds)
  29. ;
  30. ;
  31. ; Ran 1 tests containing 1 steps with 1 assertions:
  32. ; * 1 pass
  33. ;
  34. ; Stopping test component
  35. => true

If you wish to hook into another system library such as integrant,
use the ManagedSystem protocol provider in the runner namespace, which supports extension via metadata:

  1. (extend-protocol runner/ManagedSystem
  2. java.util.Map
  3. (start-system [this] (integrant/init this))
  4. (stop-system [this] (integrant/halt! this)))
  5. (runner/run-tests! (constantly {:some-system-map :with-components})
  6. tests {})
  7. ;;;Alternatively:
  8. (runner/run-tests! (constantly
  9. (with-meta {:some-system-map :with-components}
  10. {`runner/start-system (fn [this] (println "Starting test system...") this)
  11. `runner/stop-system (fn [this] (println "Stopping test system...") nil)}))
  12. tests {})

Step Inputs and Outputs

Test steps support parameterization through their inputs. Inputs can be statically
configured, pull components from the test system, or pull values built up from
previous steps in the test context.

Steps can additionally have outputs. These outputs are registered in the test
context and passed to subsequent steps. Outputs can be registered under a key,
a collection of keys, or as a function of the previous context.

  1. (defstep step-with-output-keyword
  2. "A step demonstrating a keyword output"
  3. :title "Keyword output"
  4. :test (constantly 1)
  5. :output :foo)
  6. (defstep step-with-collection-output
  7. "A step demonstrating a nested output"
  8. :title "Nested output"
  9. :test (constantly 2)
  10. :output [:a :b :c])
  11. (defstep step-with-inputs
  12. "A step demonstrating different input types"
  13. :title "Step with inputs"
  14. :inputs {;; For defaulting values
  15. :a 1
  16. ;; Extracting from test system
  17. :b (step/component ::component)
  18. ;; Extracting from test context
  19. :c (step/lookup :foo)
  20. ;; Context lookups can be a collection
  21. :d (step/lookup [:a :b :c])
  22. ;; Or for ultimate flexibility: a function
  23. :e (step/lookup (fn [ctx] (:bar ctx)))}
  24. :test (fn [{:keys [a b c d e]}]
  25. (is (every? some? [a b c d e]))))
  26. (deftest inputs-and-outputs
  27. "A test demonstrating inputs and outputs"
  28. (step-with-output-keyword)
  29. (step-with-collection-output)
  30. ;; Steps can also be defined inline
  31. #::step{:title "function output"
  32. :name 'step-with-function-output
  33. :test (constantly 3)
  34. :output (fn [ctx test-output]
  35. (assoc ctx :bar (* 2 test-output)))}
  36. (step-with-inputs
  37. ;; Override step inputs
  38. {:a :override-default}
  39. ;; Override step configuration
  40. :output :new-output
  41. :title "New Title"))
  42. (runner/run-tests!
  43. (constantly (component/system-map ::component (->TestComponent)))
  44. [(inputs-and-outputs)]
  45. {})
  46. ; Starting test system...
  47. ; Starting test component
  48. ; Running 1 tests...
  49. ;
  50. ; * Testing inputs-and-outputs
  51. ; | user:36
  52. ; | A test demonstrating inputs and outputs
  53. ; |
  54. ; +->> Keyword output
  55. ; | 0 assertions ()
  56. ; | [PASS] (0.000 seconds)
  57. ; |
  58. ; +->> Nested output
  59. ; | 0 assertions ()
  60. ; | [PASS] (0.000 seconds)
  61. ; |
  62. ; +->> Function output
  63. ; | 0 assertions ()
  64. ; | [PASS] (0.000 seconds)
  65. ; |
  66. ; +->> New Title
  67. ; | 1 assertions (1 pass)
  68. ; | [PASS] (0.000 seconds)
  69. ; |
  70. ; |
  71. ; * PASS (0.003 seconds)
  72. ;
  73. ;
  74. ; Ran 1 tests containing 4 steps with 1 assertions:
  75. ; * 1 pass
  76. ;
  77. ; Stopping test component
  78. => true

Step Cleanup

Often, steps will create some resource that should be removed on test completion.
The step/clean! multimethod along with step/register-cleanup! allows for test
steps to register resource specific cleanups. These cleanups are performed in
reverse order of registration.

  1. (defmethod step/clean! :foo/bar
  2. [system resource-type k]
  3. (printf "Removing %s resource %s" resource-type k)
  4. ;; Actual cleanup using system component
  5. )
  6. (defstep step-with-cleanup
  7. "A step demonstrating resource cleanup"
  8. :title "Step with cleanup"
  9. :test (fn [_]
  10. (step/register-cleanup! :foo/bar :my-key)))
  11. (deftest test-with-cleanup
  12. "A test demonstrating resource cleanup"
  13. (step-with-cleanup))
  14. (runner/run-tests!
  15. (constantly {})
  16. [(test-with-cleanup)]
  17. {})
  18. ; Starting test system...
  19. ; Running 1 tests...
  20. ;
  21. ; * Testing test-with-cleanup
  22. ; | user:13
  23. ; | A test demonstrating resource cleanup
  24. ; |
  25. ; +->> Step with cleanup
  26. ; | 0 assertions ()
  27. ; | [PASS] (0.000 seconds)
  28. ; |
  29. ; +->> Cleaning :foo/bar resource :my-key
  30. ; Removing :foo/bar resource :my-key |
  31. ; * PASS (0.000 seconds)
  32. ;
  33. ; Ran 1 tests containing 1 steps with 0 assertions:
  34. ; * 1 pass
  35. => true

Test Discovery

Tests can be specified explictly when running tests, or by utilizing
runner/find-tests, which retrieves tests based on a matcher. The matcher
can be either a keyword to match against test metadata or a regular expression
to match test name.

  1. (deftest ^:quick test-1
  2. ",,,"
  3. ,,,)
  4. (deftest test-2
  5. ",,,"
  6. ,,,)
  7. (runner/find-tests :quick)
  8. => (#:greenlight.test{:description ",,,", :title "test-1", :ns user, :line 1, :steps []})
  9. (runner/find-tests #"test-2")
  10. => (#:greenlight.test{:description ",,,", :title "test-2", :ns user, :line 5, :steps []})

Parallel Test Execution

Tests can be executed in parallel by providing a --parallel option
with a number of threads. Tests can be further grouped with ::test/group
metadata to indicate that tests within the same group should run serially.

If a ::test/group is not provided, a test is placed in its own group.
Groups of tests are run in parallel.

  1. (deftest simple1
  2. (math-test))
  3. (deftest simple2
  4. {::test/group :sync}
  5. (math-test))
  6. (deftest simple3
  7. {::test/group :sync}
  8. (math-test))
  9. ;; Will run `simple1` and `simple2` concurrently, then
  10. ;; `simple3` on completion of `simple2`.
  11. (runner/run-tests!
  12. (constantly {})
  13. [(simple1)
  14. (simple2)
  15. (simple3)]
  16. {:parallel 2})

Retrying steps

Integration tests can sometimes be slow, or reliant on less stable systems. When
running such tests manually, it can be helpful to not have to rerun entire tests
from the beginning. You can do this by passing {:on-fail :prompt} to the test
runner:

  1. (runner/run-tests!
  2. (constantly {})
  3. [(simple1)
  4. (simple2)
  5. (simple3)]
  6. {:on-fail :prompt})

When a step fails, it will now ask if you want to retry the step.

License

Licensed under the Apache License, Version 2.0. See the LICENSE file
for rights and restrictions.