shard-spectator/src/spectator/dsl/structure_dsl.cr

517 lines
16 KiB
Crystal
Raw Normal View History

2018-09-15 17:21:23 +00:00
require "../example_group"
module Spectator::DSL
# Domain specific language for the main structure of a spec.
# The primary components of this are `#describe`, `#context`, and `#it`.
#
# These macros define modules and classes.
# Those modules and classes are used to create the test cases.
#
# A class is created for every block of code that contains test code.
# An `#it` block creates a class derived from `RunnableExample`.
# A `#pending` block creates a class derived from `PendingExample`.
# The classes are built so that they run the example's code when invoked.
# However, the actual example code is placed into a separate "wrapper" class.
# This is done to avoid overlap with the Spectator namespace.
# The example code ends up having visibility only into itself and the DSL.
#
# Here's some skeleton code to demonstrate this:
# ```
# it "does something" do
# # Test code goes here...
# end
#
# # becomes...
#
# # Class describing the example
# # and provides a means of running the test.
# # Typically every class, module, and method
# # that the user might see or be able to reference is obscured.
# # Fresh variables from Crystal's macros are used to achive this.
# # It makes debugging Spectator more difficult,
# # but prevents name collision with user code.
# class Example123 < RunnableExample
# def initialize(group, sample_values)
# # group and sample_values are covered later.
# super
# @wrapper = Wrapper123.new(sample_values)
# end
#
# # Returns the text provided by the user.
# # This isn't stored as a member
# # so that it can be referenced directly in compiled code.
# def what
# "does something"
# end
#
# # This method is called by `RunnableExample`
# # when the example code should be ran.
# def run_instance
# @wrapper._run123
# end
# end
#
# # Wrapper class for the example code.
# # This isolates it from Spectator's internals.
# class Wrapper123
# include Context123 # More on this in a bit.
# include ExampleDSL # Include DSL for the example code.
#
# # Generated method name to avoid conflicts.
# def _run123
# # Test code goes here...
# end
# end
# ```
#
# Modules are used to provide context and share methods across examples.
# They are used as mix-ins for the example code.
# The example code wrapper class includes its parent module.
# This allows the example to access anything that was defined in the same context.
# Contexts can be nested, and this is achieved by including the parent module.
# Whenever a module or class is defined,
# it includes its parent so that functionality can be inherited.
#
# For example:
# ```
# describe "#foo" do
# subject { described_class.foo(value) }
#
# context "when given :bar" do
# let(value) { :bar }
#
# it "does something" do
# # ...
# end
# end
# end
#
# # becomes...
#
# # describe "#foo"
# module Group123
2018-11-03 18:14:29 +00:00
# # Start a new group.
# # More on this in a bit.
# Builder.start_group("#foo")
#
# def subject
# described_class.foo(value)
# end
#
# # context "when given :bar"
# module Group456
# include Group123 # Inherit parent module.
#
# # Start a nested group.
# Builder.start_group("when given :bar")
#
# def value
# :bar
# end
#
# # Wrapper class for the test case.
# class Wrapper456
# include Group456 # Include context.
#
# # Rest of example code...
# end
#
# # Example class for the test case.
# class Example456 < RunnableExample
# # Rest of example code...
# end
#
# # Add example to group.
# Builder.add_example(Example456)
#
# # End "when given :bar" group.
# Builder.end_group
# end
#
# # End "#foo" group.
# Builder.end_group
# end
# ```
#
# In addition to providing modules as mix-ins,
# example groups are defined with `#describe` and `#context`.
# The DSL makes use of `Builder` to construct the run-time portion of the spec.
# As groups are defined, they are pushed on a stack
# and popped off after everything nested in them is defined.
# `Builder` tracks the current group (top of the stack).
# This way, examples, hooks, nested groups, and other items can be added to it.
# Groups and examples are nested in a parent group.
# The only group that isn't nested is the root group - `RootExampleGroup`.
2018-11-06 21:05:41 +00:00
#
# Some example groups make use of sample values.
# Sample values are a collection of test values that can be used in examples.
# For more information, see `Internals::SampleValues`.
module StructureDSL
2018-11-06 21:05:41 +00:00
# Placeholder initializer.
# This is needed because examples and groups call `super` in their initializer.
# Those initializers pass the sample values upward through their hierarchy.
def initialize(sample_values : Internals::SampleValues)
end
2018-11-06 21:05:41 +00:00
# Creates a new example group to describe a component.
# The `what` argument describes "what" is being tested.
# Additional example groups and DSL may be nested in the block.
#
# Typically when testing a method,
# the spec is written like so:
# ```
# describe "#foo" do
# it "does something" do
# # ...
# end
# end
# ```
#
# When describing a class (or any other type),
# the `what` parameter doesn't need to be quoted.
# ```
# describe String do
# it "does something" do
# # ...
# end
# end
# ```
#
# And when combining the two together:
# ```
# describe String do
# describe "#size" do
# it "returns the length" do
# # ...
# end
# end
# end
# ```
#
# The `#describe` and `#context` are identical in terms of functionality.
# However, `#describe` is typically used on classes and methods,
# while `#context` is used for use cases and scenarios.
macro describe(what, &block)
context({{what}}) {{block}}
end
2018-09-15 17:21:23 +00:00
2018-11-06 21:05:41 +00:00
# Creates a new example group to describe a situation.
# The `what` argument describes the scenario or case being tested.
# Additional example groups and DSL may be nested in the block.
#
# The `#describe` and `#context` are identical in terms of functionality.
# However, `#describe` is typically used on classes and methods,
# while `#context` is used for use cases and scenarios.
#
# Using context blocks in conjunction with hooks, `#let`, and other methods
# provide an easy way to define the scenario in code.
# This also gives each example in the context an identical situation to run in.
#
# For instance:
# ```
# describe String do
# context "when empty" do
# subject { "" }
#
# it "has a size of zero" do
# expect(subject.size).to eq(0)
# end
#
# it "is blank" do
# expect(subject.blank?).to be_true
# end
# end
#
# context "when not empty" do
# subject { "foobar" }
#
# it "has a non-zero size" do
# expect(subject.size).to_not eq(0)
# end
#
# it "is not blank" do
# expect(subject.blank?).to be_false
# end
# end
# end
# ```
#
# While this is a somewhat contrived example,
# it demonstrates how contexts can reuse code.
# Contexts also make it clearer how a scenario is setup.
macro context(what, &block)
2018-11-06 21:05:41 +00:00
# Module for the context.
# The module uses a generated unique name.
module Group%group
2018-11-06 21:05:41 +00:00
# Include the parent module.
# Since `@type` resolves immediately,
# this will reference the parent type.
include {{@type.id}}
2018-09-15 17:21:23 +00:00
2018-11-06 21:05:41 +00:00
# Check if `what` looks like a type.
# If it is, add the `#described_class` method.
# At the time of writing this code,
# this is the way (at least that I know of)
# to check if an AST node is a type name.
#
# NOTE: In Crystal 0.27, it looks like `#resolve` can be used.
# Need to investigate, but would also increase minimum version.
{% if what.is_a?(Path) || what.is_a?(Generic) %}
_spectator_described_class {{what}}
{% end %}
2018-09-15 17:21:23 +00:00
2018-11-06 21:05:41 +00:00
# Start a new group.
::Spectator::DSL::Builder.start_group(
{{what.is_a?(StringLiteral) ? what : what.stringify}}
)
2018-11-06 21:05:41 +00:00
# Nest the block's content in the module.
{{block.body}}
2018-11-06 21:05:41 +00:00
# End the current group.
::Spectator::DSL::Builder.end_group
2018-09-15 17:21:23 +00:00
end
end
2018-09-15 17:21:23 +00:00
2018-11-07 19:26:47 +00:00
# Creates a new example group to given multiple values to.
# This method takes a collection of values
# and repeats the contents of the block with each value.
# The `collection` argument should be a literal collection,
# such as an array, or a function that returns an enumerable.
# The block should accept an argument.
# If it does, then the argument's name is used to reference
# the current item in the collection.
# If an argument isn't provided, then `#value` can be used instead.
#
# Example with a block argument:
# ```
# given some_integers do |integer|
# it "sets the value" do
# subject.value = integer
# expect(subject.value).to eq(integer)
# end
# end
# ```
#
# Same spec, but without a block argument:
# ```
# given some_integers do
# it "sets the value" do
# subject.value = value
# expect(subject.value).to eq(value)
# end
# end
# ```
#
# In the examples above, the test case (`#it` block)
# is repeated for each element in `some_integers`.
# `some_integers` is ficticous collection.
# The collection will be iterated once.
macro given(collection, &block)
2018-11-07 19:26:47 +00:00
# Figure out the name to use for the current collection element.
# If a block argument is provided, use it, otherwise use "value".
{% name = block.args.empty? ? "value".id : block.args.first %}
2018-11-07 19:26:47 +00:00
# Module for the context.
# The module uses a generated unique name.
module Group%group
2018-11-07 19:26:47 +00:00
# Include the parent module.
# Since `@type` resolves immediately,
# this will reference the parent type.
include {{@type.id}}
2018-11-07 19:26:47 +00:00
# Method for retrieving the entire collection.
# This simplifies getting the element type.
# The name is uniquely generated to prevent namespace collision.
def %collection
{{collection}}
end
2018-11-07 19:26:47 +00:00
# Value wrapper for the current element.
@%wrapper : ::Spectator::Internals::ValueWrapper
2018-11-07 19:26:47 +00:00
# Retrieves the current element from the collection.
def {{name}}
2018-11-07 19:26:47 +00:00
# Unwrap the value and return it.
# The `#first` method has a return type that matches the element type.
# So it is used on the collection method proxy to resolve the type at compile-time.
@%wrapper.as(::Spectator::Internals::TypedValueWrapper(typeof(%collection.first))).value
end
2018-11-07 19:26:47 +00:00
# Initializer to extract current element of the collection from sample values.
def initialize(sample_values : ::Spectator::Internals::SampleValues)
super
@%wrapper = sample_values.get_wrapper(:%group)
end
2018-11-07 19:26:47 +00:00
# Additional logic for setting up the collection.
# See the `#_spectator_given_collection` for nitty-gritty details.
_spectator_given_collection Collection%collection, %to_a, %collection
2018-09-15 17:21:23 +00:00
2018-11-07 19:26:47 +00:00
# Start a new example group.
# Given groups require additional configuration.
::Spectator::DSL::Builder.start_given_group(
2018-11-07 19:26:47 +00:00
{{collection.stringify}}, # String representation of the collection.
Collection%collection.new.%to_a, # All elements in the collection.
{{name.stringify}}, # Name for the current element.
:%group # Unique identifier for retrieving elements for the associated collection.
)
2018-11-07 19:26:47 +00:00
# Nest the block's content in the module.
{{block.body}}
2018-11-07 19:26:47 +00:00
# End the current group.
::Spectator::DSL::Builder.end_group
end
end
2018-09-15 17:21:23 +00:00
macro subject(&block)
let(:subject) {{block}}
end
2018-09-15 17:21:23 +00:00
macro let(name, &block)
let!(%value) {{block}}
2018-09-15 17:21:23 +00:00
@%wrapper : ::Spectator::Internals::ValueWrapper?
2018-09-15 17:21:23 +00:00
def {{name.id}}
if (wrapper = @%wrapper)
wrapper.unsafe_as(::Spectator::Internals::TypedValueWrapper(typeof(%value))).value
else
%value.tap do |value|
@%wrapper = ::Spectator::Internals::TypedValueWrapper(typeof(%value)).new(value)
2018-09-15 17:21:23 +00:00
end
end
end
end
2018-09-15 17:21:23 +00:00
macro let!(name, &block)
def {{name.id}}
{{block.body}}
2018-09-15 17:21:23 +00:00
end
end
2018-09-15 17:21:23 +00:00
macro before_all(&block)
::Spectator::DSL::Builder.add_before_all_hook {{block}}
end
2018-09-15 17:21:23 +00:00
macro before_each(&block)
::Spectator::DSL::Builder.add_before_each_hook {{block}}
end
2018-09-15 17:21:23 +00:00
macro after_all(&block)
::Spectator::DSL::Builder.add_after_all_hook {{block}}
end
2018-09-15 17:21:23 +00:00
macro after_each(&block)
::Spectator::DSL::Builder.add_after_each_hook {{block}}
end
2018-09-15 17:21:23 +00:00
macro around_each(&block)
::Spectator::DSL::Builder.add_around_each_hook {{block}}
end
2018-09-15 17:21:23 +00:00
def include_examples
raise NotImplementedError.new("Spectator::DSL#include_examples")
end
2018-09-15 17:21:23 +00:00
macro it(what, &block)
_spectator_example_wrapper(Wrapper%example, %run) {{block}}
_spectator_example(Example%example, Wrapper%example, ::Spectator::RunnableExample, {{what}}) do
protected def run_instance
@instance.%run
2018-09-15 17:21:23 +00:00
end
end
::Spectator::DSL::Builder.add_example(Example%example)
end
macro pending(what, &block)
_spectator_example_wrapper(Wrapper%example, %run) {{block}}
_spectator_example(Example%example, Wrapper%example, ::Spectator::PendingExample, {{what}})
2018-09-15 17:21:23 +00:00
::Spectator::DSL::Builder.add_example(Example%example)
end
2018-09-15 17:21:23 +00:00
macro it_behaves_like
{% raise NotImplementedError.new("it_behaves_like functionality is not implemented") %}
end
2018-09-15 17:21:23 +00:00
private macro _spectator_described_class(what)
def described_class
{{what}}.tap do |thing|
raise "#{thing} must be a type name to use #described_class or #subject,\
but it is a #{typeof(thing)}" unless thing.is_a?(Class)
end
2018-09-15 17:21:23 +00:00
end
_spectator_implicit_subject
end
private macro _spectator_implicit_subject
def subject
described_class.new
2018-09-15 17:21:23 +00:00
end
end
2018-09-23 22:07:03 +00:00
2018-11-07 19:26:47 +00:00
# :nodoc:
# Don't use this outside of Spectator DSL.
# This macro creates a class that is used to return the given collection as an array.
# This macro is *required* and cannot be embedded in the `#given` macro.
# There are a couple of reasons for this:
# 1. `@type` is resolved immediately.
# The resolution of `@type` in this macro occurs after the `#given` block.
# 2. The collection could reference a helper method or method local to the context.
# The class that the collection is defined in must have access to the context.
# Since the names are generated, and macros can't return information,
# the names must be created outside of the macro and passed in.
private macro _spectator_given_collection(class_name, to_a_method_name, collection_method_name)
2018-11-07 19:26:47 +00:00
# Class for generating an array with the collection's contents.
class {{class_name.id}}
2018-11-07 19:26:47 +00:00
# Include the parent module.
# This should be the module created for the `#given` block.
include {{@type.id}}
2018-09-23 22:07:03 +00:00
2018-11-07 19:26:47 +00:00
# Method that returns an array from the collection.
def {{to_a_method_name.id}}
{{collection_method_name.id}}.to_a
2018-09-23 22:07:03 +00:00
end
end
end
2018-09-23 22:07:03 +00:00
private macro _spectator_example_wrapper(class_name, run_method_name, &block)
class {{class_name.id}}
include ::Spectator::DSL::ExampleDSL
include {{@type.id}}
2018-09-23 22:07:03 +00:00
def initialize(sample_values : ::Spectator::Internals::SampleValues)
super
end
2018-09-23 22:07:03 +00:00
def {{run_method_name.id}}
{{block.body}}
2018-09-23 22:07:03 +00:00
end
end
end
2018-09-23 22:07:03 +00:00
private macro _spectator_example(example_class_name, wrapper_class_name, base_class, what, &block)
class {{example_class_name.id}} < {{base_class.id}}
def initialize(group : ::Spectator::ExampleGroup, sample_values : ::Spectator::Internals::SampleValues)
super
@instance = {{wrapper_class_name.id}}.new(sample_values)
end
2018-09-23 22:07:03 +00:00
{% if block.is_a?(Block) %}
{{block.body}}
{% end %}
2018-09-23 22:07:03 +00:00
def what
{{what.is_a?(StringLiteral) ? what : what.stringify}}
2018-09-23 22:07:03 +00:00
end
end
2018-09-15 17:21:23 +00:00
end
end
end