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

144 lines
3.5 KiB
Crystal
Raw Normal View History

2018-09-15 17:21:23 +00:00
require "../example_group"
module Spectator
module DSL
module StructureDSL
macro describe(what, type = "Describe", &block)
context({{what}}, {{type}}) {{block}}
end
macro context(what, type = "Context", &block)
module {{type.id}}%context
include {{@type.id}}
2018-09-15 17:21:23 +00:00
2018-09-20 00:58:44 +00:00
::Spectator::Definitions::GROUPS[\{{@type.stringify}}] =
ExampleGroup.new(
{{what.is_a?(StringLiteral) ? what : what.stringify}},
2018-09-20 00:58:44 +00:00
::Spectator::Definitions::GROUPS[{{@type.stringify}}]
)
2018-09-15 17:21:23 +00:00
2018-09-20 01:29:33 +00:00
{% if what.is_a?(Path) || what.is_a?(Generic) %}
_described_class {{what}}
{% end %}
2018-09-15 17:21:23 +00:00
{{block.body}}
end
end
macro given(collection, &block)
context({{collection}}, "Given") do
def %collection
2018-09-15 17:21:23 +00:00
{{collection}}
end
def %first
2018-09-15 17:21:23 +00:00
%collection.first
end
def {{block.args.empty? ? "value".id : block.args.first}}
nil # TODO
end
2018-09-15 17:21:23 +00:00
{{block.body}}
end
end
macro subject(&block)
let(:subject) {{block}}
end
macro let(name, &block)
let!(%value) {{block}}
@%wrapper : ValueWrapper?
def {{name.id}}
if (wrapper = @%wrapper)
wrapper.unsafe_as(TypedValueWrapper(typeof(%value))).value
else
%value.tap do |value|
@%wrapper = TypedValueWrapper(typeof(%value)).new(value)
end
end
end
end
macro let!(name, &block)
def {{name.id}}
{{block.body}}
end
end
macro before_all(&block)
2018-09-20 00:58:44 +00:00
::Spectator::Definitions::GROUPS[{{@type.stringify}}].before_all_hooks << -> {{block}}
2018-09-15 17:21:23 +00:00
end
macro before_each(&block)
2018-09-20 00:58:44 +00:00
::Spectator::Definitions::GROUPS[{{@type.stringify}}].before_each_hooks << -> {{block}}
2018-09-15 17:21:23 +00:00
end
macro after_all(&block)
2018-09-20 00:58:44 +00:00
::Spectator::Definitions::GROUPS[{{@type.stringify}}].after_all_hooks << -> {{block}}
2018-09-15 17:21:23 +00:00
end
macro after_each(&block)
2018-09-20 00:58:44 +00:00
::Spectator::Definitions::GROUPS[{{@type.stringify}}].after_each_hooks << -> {{block}}
2018-09-15 17:21:23 +00:00
end
macro around_each(&block)
2018-09-20 00:58:44 +00:00
::Spectator::Definitions::GROUPS[{{@type.stringify}}].around_each_hooks << Proc(Proc(Nil), Nil).new {{block}}
2018-09-15 17:21:23 +00:00
end
def include_examples
raise NotImplementedError.new("Spectator::DSL#include_examples")
end
macro it(description, &block)
class Example%example
include ::Spectator::DSL::ExampleDSL
include {{@type.id}}
2018-09-15 17:21:23 +00:00
def %run
2018-09-15 17:21:23 +00:00
{{block.body}}
end
end
class Wrapper%wrapper < ::Spectator::RunnableExample
2018-09-15 17:21:23 +00:00
protected def run_instance
Example%example.new.%run
2018-09-15 17:21:23 +00:00
end
def description
{{description.is_a?(StringLiteral) ? description : description.stringify}}
2018-09-15 17:21:23 +00:00
end
end
end
macro pending(description, &block)
end
2018-09-15 17:21:23 +00:00
def it_behaves_like
raise NotImplementedError.new("Spectator::DSL#it_behaves_like")
end
2018-09-15 17:21:23 +00:00
macro _described_class(what)
2018-09-20 01:29:33 +00:00
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)
2018-09-15 17:21:23 +00:00
end
2018-09-20 01:29:33 +00:00
end
2018-09-15 17:21:23 +00:00
2018-09-20 01:29:33 +00:00
_implicit_subject
2018-09-15 17:21:23 +00:00
end
macro _implicit_subject
def subject
described_class.new
end
2018-09-15 17:21:23 +00:00
end
end
end
end