Reuse iterative example group macro code

Add support for x prefix to skip sample and random_sample groups.
This commit is contained in:
Michael Miller 2021-07-17 13:25:38 -06:00
parent e506c6b981
commit 571bc7d8a5
No known key found for this signature in database
GPG key ID: FB9F12F7C646A4AD
2 changed files with 76 additions and 83 deletions

View file

@ -18,6 +18,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Add matcher to check compiled type of values. - Add matcher to check compiled type of values.
- Examples can be skipped by using a `:pending` tag. A reason method can be specified: `pending: "Some excuse"` - Examples can be skipped by using a `:pending` tag. A reason method can be specified: `pending: "Some excuse"`
- Examples can be skipped during execution by using `skip` or `pending` in the example block. - Examples can be skipped during execution by using `skip` or `pending` in the example block.
- Sample blocks can be temporarily skipped by using `xsample` or `xrandom_sample`.
### Changed ### Changed
- Simplify and reduce defined types and generics. Should speed up compilation times. - Simplify and reduce defined types and generics. Should speed up compilation times.
@ -32,6 +33,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Better error messages and detection when DSL methods are used when they shouldn't (i.e. `describe` inside `it`). - Better error messages and detection when DSL methods are used when they shouldn't (i.e. `describe` inside `it`).
- Prevent usage of reserved keywords in DSL (such as `initialize`). - Prevent usage of reserved keywords in DSL (such as `initialize`).
- The count argument for `sample` and `random_sample` groups must be named (use `count: 5` instead of just `5`). - The count argument for `sample` and `random_sample` groups must be named (use `count: 5` instead of just `5`).
- Helper methods used as arguments for `sample` and `random_sample` must be class methods.
- Other minor internal improvements and cleanup. - Other minor internal improvements and cleanup.
### Deprecated ### Deprecated

View file

@ -52,6 +52,71 @@ module Spectator::DSL
end end
end end
# Defines a macro to generate code for an iterative example group.
# The *name* is the name given to the macro.
#
# Default tags can be provided with *tags* and *metadata*.
# The tags are merged with parent groups.
# Any items with falsey values from *metadata* remove the corresponding tag.
#
# If provided, a block can be used to modify collection that will be iterated.
# It takes a single argument - the original collection from the user.
# The modified collection should be returned.
#
# TODO: Handle string interpolation in example and group names.
macro define_iterative_group(name, *tags, **metadata, &block)
macro {{name.id}}(collection, *tags, count = nil, **metadata, &block)
\{% raise "Cannot use 'sample' inside of a test block" if @def %}
class Group\%group < \{{@type.id}}
_spectator_metadata(:metadata, :super, {{tags.splat(", ")}} {{metadata.double_splat}})
_spectator_metadata(:metadata, :previous_def, \{{tags.splat(", ")}} \{{metadata.double_splat}})
def self.\%collection
\{{collection}}
end
{% if block %}
def self.%mutate({{block.args.splat}})
{{block.body}}
end
def self.\%collection
%mutate(previous_def)
end
{% end %}
\{% if count %}
def self.\%collection
previous_def.first(\{{count}})
end
\{% end %}
::Spectator::DSL::Builder.start_iterative_group(
\%collection,
\{{collection.stringify}},
\{{block.args.empty? ? :nil.id : block.args.first.stringify}},
::Spectator::Location.new(\{{block.filename}}, \{{block.line_number}}),
metadata
)
\{% if block %}
\{% if block.args.size == 1 %}
let(\{{block.args.first}}) do |example|
example.group.as(::Spectator::ExampleGroupIteration(typeof(Group\%group.\%collection.first))).item
end
\{% elsif block.args.size > 1 %}
\{% raise "Expected 1 argument for 'sample' block, but got #{block.args.size}" %}
\{% end %}
\{{block.body}}
\{% end %}
::Spectator::DSL::Builder.end_group
end
end
end
# Inserts the correct representation of a group's name. # Inserts the correct representation of a group's name.
# If *what* appears to be a type name, it will be symbolized. # If *what* appears to be a type name, it will be symbolized.
# If it's a string, then it is dropped in as-is. # If it's a string, then it is dropped in as-is.
@ -127,47 +192,10 @@ module Spectator::DSL
# #
# The number of items iterated can be restricted by specifying a *count* argument. # The number of items iterated can be restricted by specifying a *count* argument.
# The first *count* items will be used if specified, otherwise all items will be used. # The first *count* items will be used if specified, otherwise all items will be used.
# define_iterative_group :sample
# TODO: Handle string interpolation in example and group names.
macro sample(collection, *tags, count = nil, **metadata, &block)
{% raise "Cannot use 'sample' inside of a test block" if @def %}
class Group%group < {{@type.id}} # :ditto:
_spectator_metadata(:metadata, :super, {{tags.splat(", ")}} {{metadata.double_splat}}) define_iterative_group :xsample, skip: "Temporarily skipped with xsample"
def self.%collection
{{collection}}
end
{% if count %}
def self.%collection
previous_def.first({{count}})
end
{% end %}
::Spectator::DSL::Builder.start_iterative_group(
%collection,
{{collection.stringify}},
{{block.args.empty? ? :nil.id : block.args.first.stringify}},
::Spectator::Location.new({{block.filename}}, {{block.line_number}}),
metadata
)
{% if block %}
{% if block.args.size == 1 %}
let({{block.args.first}}) do |example|
example.group.as(::Spectator::ExampleGroupIteration(typeof(Group%group.%collection.first))).item
end
{% elsif block.args.size > 1 %}
{% raise "Expected 1 argument for 'sample' block, but got #{block.args.size}" %}
{% end %}
{{block.body}}
{% end %}
::Spectator::DSL::Builder.end_group
end
end
# Defines a new iterative example group. # Defines a new iterative example group.
# This type of group duplicates its contents for each element in *collection*. # This type of group duplicates its contents for each element in *collection*.
@ -182,50 +210,13 @@ module Spectator::DSL
# #
# The number of items iterated can be restricted by specifying a *count* argument. # The number of items iterated can be restricted by specifying a *count* argument.
# The first *count* items will be used if specified, otherwise all items will be used. # The first *count* items will be used if specified, otherwise all items will be used.
# define_iterative_group :random_sample do |collection|
# TODO: Handle string interpolation in example and group names. collection.to_a.shuffle(::Spectator.random)
macro random_sample(collection, *tags, count = nil, **metadata, &block)
{% raise "Cannot use 'sample' inside of a test block" if @def %}
class Group%group < {{@type.id}}
_spectator_metadata(:metadata, :super, {{tags.splat(", ")}} {{metadata.double_splat}})
def self.%collection
{{collection}}
end end
{% if count %} # :ditto:
def self.%collection define_iterative_group :xrandom_sample, skip: "Temporarily skipped with xrandom_sample" do |collection|
previous_def.sample({{count}}, ::Spectator.random) collection.to_a.shuffle(::Spectator.random)
end
{% else %}
def self.%collection
previous_def.to_a.shuffle(::Spectator.random)
end
{% end %}
::Spectator::DSL::Builder.start_iterative_group(
%collection,
{{collection.stringify}},
{{block.args.empty? ? :nil.id : block.args.first.stringify}},
::Spectator::Location.new({{block.filename}}, {{block.line_number}}),
metadata
)
{% if block %}
{% if block.args.size == 1 %}
let({{block.args.first}}) do |example|
example.group.as(::Spectator::ExampleGroupIteration(typeof(Group%group.%collection.first))).item
end
{% elsif block.args.size > 1 %}
{% raise "Expected 1 argument for 'sample' block, but got #{block.args.size}" %}
{% end %}
{{block.body}}
{% end %}
::Spectator::DSL::Builder.end_group
end
end end
end end
end end