mutant 0.8.24 → 0.9.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.circleci/config.yml +3 -3
- data/Changelog.md +14 -654
- data/Gemfile +13 -0
- data/Gemfile.lock +59 -64
- data/LICENSE +271 -20
- data/README.md +73 -140
- data/Rakefile +0 -21
- data/bin/mutant +7 -2
- data/config/reek.yml +2 -1
- data/config/rubocop.yml +5 -9
- data/docs/incremental.md +76 -0
- data/docs/known-problems.md +0 -14
- data/docs/mutant-minitest.md +1 -1
- data/docs/mutant-rspec.md +2 -24
- data/lib/mutant.rb +45 -53
- data/lib/mutant/ast/nodes.rb +0 -2
- data/lib/mutant/ast/types.rb +1 -117
- data/lib/mutant/base.rb +192 -0
- data/lib/mutant/bootstrap.rb +145 -0
- data/lib/mutant/cli.rb +68 -54
- data/lib/mutant/config.rb +119 -6
- data/lib/mutant/env.rb +94 -8
- data/lib/mutant/expression.rb +6 -1
- data/lib/mutant/expression/parser.rb +9 -31
- data/lib/mutant/integration.rb +64 -36
- data/lib/mutant/isolation.rb +16 -1
- data/lib/mutant/isolation/fork.rb +105 -40
- data/lib/mutant/license.rb +34 -0
- data/lib/mutant/license/subscription.rb +47 -0
- data/lib/mutant/license/subscription/commercial.rb +57 -0
- data/lib/mutant/license/subscription/opensource.rb +77 -0
- data/lib/mutant/loader.rb +27 -4
- data/lib/mutant/matcher.rb +48 -1
- data/lib/mutant/matcher/chain.rb +1 -1
- data/lib/mutant/matcher/config.rb +0 -2
- data/lib/mutant/matcher/filter.rb +1 -1
- data/lib/mutant/matcher/method.rb +11 -7
- data/lib/mutant/matcher/methods.rb +1 -1
- data/lib/mutant/matcher/namespace.rb +1 -1
- data/lib/mutant/matcher/null.rb +1 -1
- data/lib/mutant/matcher/scope.rb +1 -1
- data/lib/mutant/meta/example/dsl.rb +0 -8
- data/lib/mutant/mutation.rb +1 -2
- data/lib/mutant/mutator/node.rb +2 -9
- data/lib/mutant/mutator/node/arguments.rb +1 -1
- data/lib/mutant/mutator/node/class.rb +0 -8
- data/lib/mutant/mutator/node/define.rb +0 -12
- data/lib/mutant/mutator/node/generic.rb +30 -44
- data/lib/mutant/mutator/node/index.rb +4 -4
- data/lib/mutant/mutator/node/literal/regex.rb +0 -39
- data/lib/mutant/mutator/node/send.rb +13 -12
- data/lib/mutant/parallel.rb +61 -40
- data/lib/mutant/parallel/driver.rb +59 -0
- data/lib/mutant/parallel/source.rb +6 -2
- data/lib/mutant/parallel/worker.rb +63 -45
- data/lib/mutant/range.rb +15 -0
- data/lib/mutant/reporter/cli.rb +5 -11
- data/lib/mutant/reporter/cli/format.rb +3 -46
- data/lib/mutant/reporter/cli/printer/config.rb +5 -6
- data/lib/mutant/reporter/cli/printer/env.rb +40 -0
- data/lib/mutant/reporter/cli/printer/env_progress.rb +13 -17
- data/lib/mutant/reporter/cli/printer/isolation_result.rb +17 -3
- data/lib/mutant/reporter/cli/printer/mutation_result.rb +2 -3
- data/lib/mutant/reporter/cli/printer/status_progressive.rb +19 -10
- data/lib/mutant/repository.rb +0 -65
- data/lib/mutant/repository/diff.rb +104 -0
- data/lib/mutant/repository/diff/ranges.rb +52 -0
- data/lib/mutant/result.rb +16 -7
- data/lib/mutant/runner.rb +38 -47
- data/lib/mutant/runner/sink.rb +1 -1
- data/lib/mutant/selector/null.rb +19 -0
- data/lib/mutant/subject.rb +3 -1
- data/lib/mutant/subject/method/instance.rb +3 -1
- data/lib/mutant/transform.rb +511 -0
- data/lib/mutant/variable.rb +282 -0
- data/lib/mutant/version.rb +1 -1
- data/lib/mutant/warnings.rb +113 -0
- data/meta/case.rb +1 -0
- data/meta/class.rb +0 -9
- data/meta/def.rb +1 -26
- data/meta/regexp.rb +10 -20
- data/meta/send.rb +14 -46
- data/mutant-minitest.gemspec +1 -1
- data/mutant-rspec.gemspec +2 -2
- data/mutant.gemspec +15 -16
- data/mutant.yml +6 -0
- data/spec/integration/mutant/isolation/fork_spec.rb +22 -5
- data/spec/integration/mutant/minitest_spec.rb +3 -2
- data/spec/integration/mutant/rspec_spec.rb +4 -3
- data/spec/integrations.yml +16 -13
- data/spec/shared/base_behavior.rb +45 -0
- data/spec/shared/framework_integration_behavior.rb +43 -14
- data/spec/spec_helper.rb +21 -17
- data/spec/support/corpus.rb +56 -95
- data/spec/support/shared_context.rb +37 -14
- data/spec/support/xspec.rb +7 -3
- data/spec/unit/mutant/bootstrap_spec.rb +216 -0
- data/spec/unit/mutant/cli_spec.rb +173 -117
- data/spec/unit/mutant/config_spec.rb +126 -0
- data/spec/unit/mutant/either_spec.rb +247 -0
- data/spec/unit/mutant/env_spec.rb +162 -40
- data/spec/unit/mutant/expression/method_spec.rb +16 -0
- data/spec/unit/mutant/expression/parser_spec.rb +29 -33
- data/spec/unit/mutant/expression_spec.rb +5 -7
- data/spec/unit/mutant/integration_spec.rb +100 -9
- data/spec/unit/mutant/isolation/fork_spec.rb +125 -67
- data/spec/unit/mutant/isolation/result_spec.rb +33 -1
- data/spec/unit/mutant/license_spec.rb +257 -0
- data/spec/unit/mutant/loader_spec.rb +50 -11
- data/spec/unit/mutant/matcher/compiler_spec.rb +0 -78
- data/spec/unit/mutant/matcher/method/instance_spec.rb +55 -11
- data/spec/unit/mutant/matcher/method/singleton_spec.rb +12 -2
- data/spec/unit/mutant/matcher_spec.rb +102 -0
- data/spec/unit/mutant/maybe_spec.rb +60 -0
- data/spec/unit/mutant/meta/example/dsl_spec.rb +1 -17
- data/spec/unit/mutant/mutation_spec.rb +13 -6
- data/spec/unit/mutant/parallel/driver_spec.rb +112 -14
- data/spec/unit/mutant/parallel/source/array_spec.rb +25 -17
- data/spec/unit/mutant/parallel/worker_spec.rb +182 -44
- data/spec/unit/mutant/parallel_spec.rb +105 -8
- data/spec/unit/mutant/range_spec.rb +141 -0
- data/spec/unit/mutant/reporter/cli/printer/config_spec.rb +7 -21
- data/spec/unit/mutant/reporter/cli/printer/env_progress_spec.rb +15 -6
- data/spec/unit/mutant/reporter/cli/printer/env_result_spec.rb +10 -2
- data/spec/unit/mutant/reporter/cli/printer/isolation_result_spec.rb +12 -4
- data/spec/unit/mutant/reporter/cli/printer/mutation_result_spec.rb +31 -2
- data/spec/unit/mutant/reporter/cli/printer/status_progressive_spec.rb +4 -4
- data/spec/unit/mutant/reporter/cli/printer/subject_result_spec.rb +5 -0
- data/spec/unit/mutant/reporter/cli_spec.rb +46 -123
- data/spec/unit/mutant/repository/diff/ranges_spec.rb +180 -0
- data/spec/unit/mutant/repository/diff_spec.rb +84 -71
- data/spec/unit/mutant/require_highjack_spec.rb +1 -1
- data/spec/unit/mutant/result/env_spec.rb +39 -9
- data/spec/unit/mutant/result/test_spec.rb +14 -0
- data/spec/unit/mutant/runner_spec.rb +88 -41
- data/spec/unit/mutant/selector/expression_spec.rb +11 -10
- data/spec/unit/mutant/selector/null_spec.rb +17 -0
- data/spec/unit/mutant/subject/method/instance_spec.rb +44 -5
- data/spec/unit/mutant/subject/method/singleton_spec.rb +9 -2
- data/spec/unit/mutant/subject_spec.rb +9 -1
- data/spec/unit/mutant/transform/array_spec.rb +92 -0
- data/spec/unit/mutant/transform/bool_spec.rb +63 -0
- data/spec/unit/mutant/transform/error_spec.rb +132 -0
- data/spec/unit/mutant/transform/exception_spec.rb +44 -0
- data/spec/unit/mutant/transform/hash_spec.rb +236 -0
- data/spec/unit/mutant/transform/index_spec.rb +92 -0
- data/spec/unit/mutant/transform/named_spec.rb +49 -0
- data/spec/unit/mutant/transform/primitive_spec.rb +56 -0
- data/spec/unit/mutant/transform/sequence_spec.rb +98 -0
- data/spec/unit/mutant/variable_spec.rb +618 -0
- data/spec/unit/mutant/warnings_spec.rb +89 -0
- data/spec/unit/mutant/world_spec.rb +63 -0
- data/test_app/Gemfile.minitest +0 -2
- metadata +79 -113
- data/.gitattributes +0 -1
- data/.ruby-gemset +0 -1
- data/config/triage.yml +0 -2
- data/lib/mutant/actor.rb +0 -57
- data/lib/mutant/actor/env.rb +0 -31
- data/lib/mutant/actor/mailbox.rb +0 -34
- data/lib/mutant/actor/receiver.rb +0 -42
- data/lib/mutant/actor/sender.rb +0 -26
- data/lib/mutant/ast/meta/restarg.rb +0 -19
- data/lib/mutant/ast/regexp.rb +0 -42
- data/lib/mutant/ast/regexp/transformer.rb +0 -187
- data/lib/mutant/ast/regexp/transformer/direct.rb +0 -123
- data/lib/mutant/ast/regexp/transformer/named_group.rb +0 -59
- data/lib/mutant/ast/regexp/transformer/options_group.rb +0 -83
- data/lib/mutant/ast/regexp/transformer/quantifier.rb +0 -114
- data/lib/mutant/ast/regexp/transformer/recursive.rb +0 -58
- data/lib/mutant/ast/regexp/transformer/root.rb +0 -31
- data/lib/mutant/ast/regexp/transformer/text.rb +0 -60
- data/lib/mutant/env/bootstrap.rb +0 -160
- data/lib/mutant/matcher/compiler.rb +0 -60
- data/lib/mutant/mutator/node/regexp.rb +0 -35
- data/lib/mutant/mutator/node/regexp/alternation_meta.rb +0 -23
- data/lib/mutant/mutator/node/regexp/capture_group.rb +0 -28
- data/lib/mutant/mutator/node/regexp/character_type.rb +0 -32
- data/lib/mutant/mutator/node/regexp/end_of_line_anchor.rb +0 -23
- data/lib/mutant/mutator/node/regexp/end_of_string_or_before_end_of_line_anchor.rb +0 -23
- data/lib/mutant/mutator/node/regexp/greedy_zero_or_more.rb +0 -27
- data/lib/mutant/parallel/master.rb +0 -181
- data/lib/mutant/reporter/cli/printer/status.rb +0 -53
- data/lib/mutant/reporter/cli/tput.rb +0 -46
- data/lib/mutant/warning_filter.rb +0 -61
- data/meta/regexp/character_types.rb +0 -23
- data/meta/regexp/regexp_alternation_meta.rb +0 -13
- data/meta/regexp/regexp_bol_anchor.rb +0 -10
- data/meta/regexp/regexp_bos_anchor.rb +0 -18
- data/meta/regexp/regexp_capture_group.rb +0 -19
- data/meta/regexp/regexp_eol_anchor.rb +0 -10
- data/meta/regexp/regexp_eos_anchor.rb +0 -8
- data/meta/regexp/regexp_eos_ob_eol_anchor.rb +0 -10
- data/meta/regexp/regexp_greedy_zero_or_more.rb +0 -12
- data/meta/regexp/regexp_root_expression.rb +0 -10
- data/meta/restarg.rb +0 -10
- data/spec/support/fake_actor.rb +0 -111
- data/spec/support/warning.rb +0 -66
- data/spec/unit/mutant/actor/binding_spec.rb +0 -34
- data/spec/unit/mutant/actor/env_spec.rb +0 -31
- data/spec/unit/mutant/actor/mailbox_spec.rb +0 -28
- data/spec/unit/mutant/actor/message_spec.rb +0 -25
- data/spec/unit/mutant/actor/receiver_spec.rb +0 -58
- data/spec/unit/mutant/actor/sender_spec.rb +0 -24
- data/spec/unit/mutant/ast/regexp/parse_spec.rb +0 -19
- data/spec/unit/mutant/ast/regexp/transformer/lookup_table/table_spec.rb +0 -21
- data/spec/unit/mutant/ast/regexp/transformer/lookup_table_spec.rb +0 -35
- data/spec/unit/mutant/ast/regexp/transformer_spec.rb +0 -21
- data/spec/unit/mutant/ast/regexp_spec.rb +0 -704
- data/spec/unit/mutant/env/bootstrap_spec.rb +0 -188
- data/spec/unit/mutant/matcher/compiler/subject_prefix_spec.rb +0 -26
- data/spec/unit/mutant/parallel/master_spec.rb +0 -338
- data/spec/unit/mutant/reporter/cli/printer/status_spec.rb +0 -121
- data/spec/unit/mutant/reporter/cli/tput_spec.rb +0 -50
- data/spec/unit/mutant/warning_filter_spec.rb +0 -106
- data/spec/unit/mutant_spec.rb +0 -17
- data/test_app/Gemfile.rspec3.7 +0 -7
@@ -1,60 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
module AST
|
5
|
-
module Regexp
|
6
|
-
class Transformer
|
7
|
-
# Regexp AST transformer for nodes that encode a text value
|
8
|
-
class Text < self
|
9
|
-
# Mapper from `Regexp::Expression` to `Parser::AST::Node`
|
10
|
-
class ExpressionToAST < Transformer::ExpressionToAST
|
11
|
-
# Transform expression into node preserving text value
|
12
|
-
#
|
13
|
-
# @return [Parser::AST::Node]
|
14
|
-
def call
|
15
|
-
quantify(ast(expression.text))
|
16
|
-
end
|
17
|
-
end # ExpressionToAST
|
18
|
-
|
19
|
-
# Mapper from `Parser::AST::Node` to `Regexp::Expression`
|
20
|
-
class ASTToExpression < Transformer::ASTToExpression
|
21
|
-
include LookupTable
|
22
|
-
|
23
|
-
# rubocop:disable LineLength
|
24
|
-
TABLE = Table.create(
|
25
|
-
[:regexp_literal_literal, %i[literal literal], ::Regexp::Expression::Literal],
|
26
|
-
[:regexp_comment_group, %i[group comment], ::Regexp::Expression::Group::Comment],
|
27
|
-
[:regexp_number_backref, %i[backref number], ::Regexp::Expression::Backreference::Number],
|
28
|
-
[:regexp_name_call_backref, %i[backref name_call], ::Regexp::Expression::Backreference::NameCall],
|
29
|
-
[:regexp_whitespace_free_space, %i[free_space whitespace], ::Regexp::Expression::WhiteSpace],
|
30
|
-
[:regexp_comment_free_space, %i[free_space comment], ::Regexp::Expression::WhiteSpace],
|
31
|
-
[:regexp_hex_escape, %i[escape hex], ::Regexp::Expression::EscapeSequence::Hex],
|
32
|
-
[:regexp_octal_escape, %i[escape octal], ::Regexp::Expression::EscapeSequence::Octal],
|
33
|
-
[:regexp_literal_escape, %i[escape literal], ::Regexp::Expression::EscapeSequence::Literal],
|
34
|
-
[:regexp_backslash_escape, %i[escape backslash], ::Regexp::Expression::EscapeSequence::Literal],
|
35
|
-
[:regexp_tab_escape, %i[escape tab], ::Regexp::Expression::EscapeSequence::Literal],
|
36
|
-
[:regexp_codepoint_list_escape, %i[escape codepoint_list], ::Regexp::Expression::EscapeSequence::CodepointList],
|
37
|
-
[:regexp_codepoint_escape, %i[escape codepoint], ::Regexp::Expression::EscapeSequence::Codepoint],
|
38
|
-
[:regexp_control_escape, %i[escape control], ::Regexp::Expression::EscapeSequence::Control],
|
39
|
-
[:regexp_meta_sequence_escape, %i[escape meta_sequence], ::Regexp::Expression::EscapeSequence::Control],
|
40
|
-
[:regexp_condition_conditional, %i[conditional condition], ::Regexp::Expression::Conditional::Condition]
|
41
|
-
)
|
42
|
-
|
43
|
-
private
|
44
|
-
|
45
|
-
# Transform node to expression with text value
|
46
|
-
#
|
47
|
-
# @return [Regexp::Expression]
|
48
|
-
def transform
|
49
|
-
token = expression_token.dup
|
50
|
-
token.text = Util.one(node.children)
|
51
|
-
expression_class.new(token)
|
52
|
-
end
|
53
|
-
end # ASTToExpression
|
54
|
-
|
55
|
-
ASTToExpression::TABLE.types.each(&method(:register))
|
56
|
-
end # Text
|
57
|
-
end # Transformer
|
58
|
-
end # Regexp
|
59
|
-
end # AST
|
60
|
-
end # Mutant
|
data/lib/mutant/env/bootstrap.rb
DELETED
@@ -1,160 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Env
|
5
|
-
# Bootstrap environment
|
6
|
-
class Bootstrap
|
7
|
-
include Adamantium::Flat, Concord::Public.new(:config), Procto.call(:env)
|
8
|
-
|
9
|
-
SEMANTICS_MESSAGE_FORMAT =
|
10
|
-
"%<message>s. Fix your lib to follow normal ruby semantics!\n" \
|
11
|
-
'{Module,Class}#name should return resolvable constant name as String or nil'
|
12
|
-
|
13
|
-
CLASS_NAME_RAISED_EXCEPTION =
|
14
|
-
'%<scope_class>s#name from: %<scope>s raised an error: %<exception>s'
|
15
|
-
|
16
|
-
CLASS_NAME_TYPE_MISMATCH_FORMAT =
|
17
|
-
'%<scope_class>s#name from: %<scope>s returned %<name>s'
|
18
|
-
|
19
|
-
private_constant(*constants(false))
|
20
|
-
|
21
|
-
# Scopes that are eligible for matching
|
22
|
-
#
|
23
|
-
# @return [Enumerable<Matcher::Scope>]
|
24
|
-
attr_reader :matchable_scopes
|
25
|
-
|
26
|
-
# Parser for this environment
|
27
|
-
#
|
28
|
-
# @return [Parser]
|
29
|
-
attr_reader :parser
|
30
|
-
|
31
|
-
# Initialize object
|
32
|
-
#
|
33
|
-
# @return [Object]
|
34
|
-
def initialize(*)
|
35
|
-
super
|
36
|
-
@parser = Parser.new
|
37
|
-
infect
|
38
|
-
initialize_matchable_scopes
|
39
|
-
@integration = config.integration.new(config).setup
|
40
|
-
end
|
41
|
-
|
42
|
-
# Print warning message
|
43
|
-
#
|
44
|
-
# @param [String]
|
45
|
-
#
|
46
|
-
# @return [self]
|
47
|
-
def warn(message)
|
48
|
-
config.reporter.warn(message)
|
49
|
-
self
|
50
|
-
end
|
51
|
-
|
52
|
-
# Environment after bootstraping
|
53
|
-
#
|
54
|
-
# @return [Env]
|
55
|
-
# rubocop:disable MethodLength
|
56
|
-
#
|
57
|
-
def env
|
58
|
-
subjects = matched_subjects
|
59
|
-
Env.new(
|
60
|
-
actor_env: Actor::Env.new(Thread),
|
61
|
-
config: config,
|
62
|
-
integration: integration,
|
63
|
-
matchable_scopes: matchable_scopes,
|
64
|
-
mutations: subjects.flat_map(&:mutations),
|
65
|
-
parser: parser,
|
66
|
-
selector: Selector::Expression.new(integration),
|
67
|
-
subjects: subjects
|
68
|
-
)
|
69
|
-
end
|
70
|
-
|
71
|
-
private
|
72
|
-
|
73
|
-
# Configured mutant integration
|
74
|
-
#
|
75
|
-
# @return [Mutant::Integration]
|
76
|
-
attr_reader :integration
|
77
|
-
|
78
|
-
# Scope name from scoping object
|
79
|
-
#
|
80
|
-
# @param [Class, Module] scope
|
81
|
-
#
|
82
|
-
# @return [String]
|
83
|
-
# if scope has a name and does not raise exceptions obtaining it
|
84
|
-
#
|
85
|
-
# @return [nil]
|
86
|
-
# otherwise
|
87
|
-
def scope_name(scope)
|
88
|
-
scope.name
|
89
|
-
rescue => exception
|
90
|
-
semantics_warning(
|
91
|
-
CLASS_NAME_RAISED_EXCEPTION,
|
92
|
-
exception: exception.inspect,
|
93
|
-
scope: scope,
|
94
|
-
scope_class: scope.class
|
95
|
-
)
|
96
|
-
nil
|
97
|
-
end
|
98
|
-
|
99
|
-
# Infect environment
|
100
|
-
#
|
101
|
-
# @return [undefined]
|
102
|
-
def infect
|
103
|
-
config.includes.each(&config.load_path.method(:<<))
|
104
|
-
config.requires.each(&config.kernel.method(:require))
|
105
|
-
end
|
106
|
-
|
107
|
-
# Matched subjects
|
108
|
-
#
|
109
|
-
# @return [Enumerable<Subject>]
|
110
|
-
def matched_subjects
|
111
|
-
Matcher::Compiler.call(config.matcher).call(self)
|
112
|
-
end
|
113
|
-
|
114
|
-
# Initialize matchable scopes
|
115
|
-
#
|
116
|
-
# @return [undefined]
|
117
|
-
def initialize_matchable_scopes
|
118
|
-
scopes = ObjectSpace.each_object(Module).each_with_object([]) do |scope, aggregate|
|
119
|
-
expression = expression(scope) || next
|
120
|
-
aggregate << Scope.new(scope, expression)
|
121
|
-
end
|
122
|
-
|
123
|
-
@matchable_scopes = scopes.sort_by { |scope| scope.expression.syntax }
|
124
|
-
end
|
125
|
-
|
126
|
-
# Try to turn scope into expression
|
127
|
-
#
|
128
|
-
# @param [Class, Module] scope
|
129
|
-
#
|
130
|
-
# @return [Expression]
|
131
|
-
# if scope can be represented in an expression
|
132
|
-
#
|
133
|
-
# @return [nil]
|
134
|
-
# otherwise
|
135
|
-
def expression(scope)
|
136
|
-
name = scope_name(scope) or return
|
137
|
-
|
138
|
-
unless name.instance_of?(String)
|
139
|
-
semantics_warning(
|
140
|
-
CLASS_NAME_TYPE_MISMATCH_FORMAT,
|
141
|
-
name: name,
|
142
|
-
scope_class: scope.class,
|
143
|
-
scope: scope
|
144
|
-
)
|
145
|
-
return
|
146
|
-
end
|
147
|
-
|
148
|
-
config.expression_parser.try_parse(name)
|
149
|
-
end
|
150
|
-
|
151
|
-
# Write a semantics warning
|
152
|
-
#
|
153
|
-
# @return [undefined]
|
154
|
-
def semantics_warning(format, options)
|
155
|
-
message = format % options
|
156
|
-
warn(SEMANTICS_MESSAGE_FORMAT % { message: message })
|
157
|
-
end
|
158
|
-
end # Bootstrap
|
159
|
-
end # Env
|
160
|
-
end # Mutant
|
@@ -1,60 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Matcher
|
5
|
-
|
6
|
-
# Compiler for complex matchers
|
7
|
-
class Compiler
|
8
|
-
include Concord.new(:config), AST::Sexp, Procto.call(:result)
|
9
|
-
|
10
|
-
# Generated matcher
|
11
|
-
#
|
12
|
-
# @return [Matcher]
|
13
|
-
def result
|
14
|
-
Filter.new(
|
15
|
-
Chain.new(config.match_expressions.map(&:matcher)),
|
16
|
-
Morpher::Evaluator::Predicate::Boolean::And.new(
|
17
|
-
[
|
18
|
-
ignored_subjects,
|
19
|
-
filtered_subjects
|
20
|
-
]
|
21
|
-
)
|
22
|
-
)
|
23
|
-
end
|
24
|
-
|
25
|
-
# Subject expression prefix predicate
|
26
|
-
class SubjectPrefix
|
27
|
-
include Concord.new(:expression)
|
28
|
-
|
29
|
-
# Test if subject expression is matched by prefix
|
30
|
-
#
|
31
|
-
# @return [Boolean]
|
32
|
-
def call(subject)
|
33
|
-
expression.prefix?(subject.expression)
|
34
|
-
end
|
35
|
-
|
36
|
-
end # SubjectPrefix
|
37
|
-
|
38
|
-
private
|
39
|
-
|
40
|
-
# Predicate returning false on expression ignored subject
|
41
|
-
#
|
42
|
-
# @return [#call]
|
43
|
-
def ignored_subjects
|
44
|
-
Morpher::Evaluator::Predicate::Boolean::Negation.new(
|
45
|
-
Morpher::Evaluator::Predicate::Boolean::Or.new(
|
46
|
-
config.ignore_expressions.map(&SubjectPrefix.method(:new))
|
47
|
-
)
|
48
|
-
)
|
49
|
-
end
|
50
|
-
|
51
|
-
# Predicate returning false on filtered subject
|
52
|
-
#
|
53
|
-
# @return [#call]
|
54
|
-
def filtered_subjects
|
55
|
-
Morpher::Evaluator::Predicate::Boolean::And.new(config.subject_filters)
|
56
|
-
end
|
57
|
-
|
58
|
-
end # Compiler
|
59
|
-
end # Matcher
|
60
|
-
end # Mutant
|
@@ -1,35 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Mutator
|
5
|
-
class Node
|
6
|
-
module Regexp
|
7
|
-
# Mutator for root expression regexp wrapper
|
8
|
-
class RootExpression < Node
|
9
|
-
handle(:regexp_root_expression)
|
10
|
-
|
11
|
-
# Emit mutations for children of root node
|
12
|
-
#
|
13
|
-
# @return [undefined]
|
14
|
-
def dispatch
|
15
|
-
children.each_index(&method(:mutate_child))
|
16
|
-
end
|
17
|
-
end # RootExpression
|
18
|
-
|
19
|
-
# Mutator for beginning of line anchor `^`
|
20
|
-
class BeginningOfLineAnchor < Node
|
21
|
-
handle(:regexp_bol_anchor)
|
22
|
-
|
23
|
-
# Emit mutations
|
24
|
-
#
|
25
|
-
# Replace `^` with `\A`
|
26
|
-
#
|
27
|
-
# @return [undefined]
|
28
|
-
def dispatch
|
29
|
-
emit(s(:regexp_bos_anchor))
|
30
|
-
end
|
31
|
-
end # BeginningOfLineAnchor
|
32
|
-
end # Regexp
|
33
|
-
end # Node
|
34
|
-
end # Mutator
|
35
|
-
end # Mutant
|
@@ -1,23 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Mutator
|
5
|
-
class Node
|
6
|
-
module Regexp
|
7
|
-
# Mutator for pipe in `/foo|bar/` regexp
|
8
|
-
class AlternationMeta < Node
|
9
|
-
handle(:regexp_alternation_meta)
|
10
|
-
|
11
|
-
private
|
12
|
-
|
13
|
-
# Dispatch mutations
|
14
|
-
#
|
15
|
-
# @return [undefined]
|
16
|
-
def dispatch
|
17
|
-
children.each_index(&method(:delete_child))
|
18
|
-
end
|
19
|
-
end # AlternationMeta
|
20
|
-
end # Regexp
|
21
|
-
end # Node
|
22
|
-
end # Mutator
|
23
|
-
end # Mutant
|
@@ -1,28 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Mutator
|
5
|
-
class Node
|
6
|
-
module Regexp
|
7
|
-
# Mutator for regexp capture groups, such as `/(foo)/`
|
8
|
-
class CaptureGroup < Node
|
9
|
-
handle(:regexp_capture_group)
|
10
|
-
|
11
|
-
children :group
|
12
|
-
|
13
|
-
# Emit mutations
|
14
|
-
#
|
15
|
-
# Replace `(captured_group)` with `(?:non_captured_group)`
|
16
|
-
#
|
17
|
-
# @return [undefined]
|
18
|
-
def dispatch
|
19
|
-
return unless group
|
20
|
-
|
21
|
-
emit(s(:regexp_passive_group, group))
|
22
|
-
emit_group_mutations
|
23
|
-
end
|
24
|
-
end # EndOfLineAnchor
|
25
|
-
end # Regexp
|
26
|
-
end # Node
|
27
|
-
end # Mutator
|
28
|
-
end # Mutant
|
@@ -1,32 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Mutator
|
5
|
-
class Node
|
6
|
-
module Regexp
|
7
|
-
# Character type mutator
|
8
|
-
class CharacterType < Node
|
9
|
-
map = {
|
10
|
-
regexp_digit_type: :regexp_nondigit_type,
|
11
|
-
regexp_hex_type: :regexp_nonhex_type,
|
12
|
-
regexp_space_type: :regexp_nonspace_type,
|
13
|
-
regexp_word_boundary_anchor: :regexp_nonword_boundary_anchor,
|
14
|
-
regexp_word_type: :regexp_nonword_type,
|
15
|
-
regexp_xgrapheme_type: :regexp_linebreak_type
|
16
|
-
}
|
17
|
-
|
18
|
-
MAP = IceNine.deep_freeze(map.merge(map.invert))
|
19
|
-
|
20
|
-
handle(*MAP.keys)
|
21
|
-
|
22
|
-
# Mutate to invert character type
|
23
|
-
#
|
24
|
-
# @return [undefined]
|
25
|
-
def dispatch
|
26
|
-
emit(s(MAP.fetch(node.type)))
|
27
|
-
end
|
28
|
-
end # CharacterType
|
29
|
-
end # Regexp
|
30
|
-
end # Node
|
31
|
-
end # Mutator
|
32
|
-
end # Mutant
|
@@ -1,23 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Mutator
|
5
|
-
class Node
|
6
|
-
module Regexp
|
7
|
-
# Mutator for end of line anchor `$`
|
8
|
-
class EndOfLineAnchor < Node
|
9
|
-
handle(:regexp_eol_anchor)
|
10
|
-
|
11
|
-
# Emit mutations
|
12
|
-
#
|
13
|
-
# Replace `$` with `\z`
|
14
|
-
#
|
15
|
-
# @return [undefined]
|
16
|
-
def dispatch
|
17
|
-
emit(s(:regexp_eos_anchor))
|
18
|
-
end
|
19
|
-
end # EndOfLineAnchor
|
20
|
-
end # Regexp
|
21
|
-
end # Node
|
22
|
-
end # Mutator
|
23
|
-
end # Mutant
|
@@ -1,23 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module Mutant
|
4
|
-
class Mutator
|
5
|
-
class Node
|
6
|
-
module Regexp
|
7
|
-
# Mutator for end of line or before end of string anchor `\Z`
|
8
|
-
class EndOfStringOrBeforeEndOfLineAnchor < Node
|
9
|
-
handle(:regexp_eos_ob_eol_anchor)
|
10
|
-
|
11
|
-
# Emit mutations
|
12
|
-
#
|
13
|
-
# Replace `\Z` with `\z`
|
14
|
-
#
|
15
|
-
# @return [undefined]
|
16
|
-
def dispatch
|
17
|
-
emit(s(:regexp_eos_anchor))
|
18
|
-
end
|
19
|
-
end # EndOfStringOrBeforeEndOfLineAnchor
|
20
|
-
end # Regexp
|
21
|
-
end # Node
|
22
|
-
end # Mutator
|
23
|
-
end # Mutant
|