mutant 0.10.6 → 0.10.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/mutant.rb +4 -3
- data/lib/mutant/cli/command/run.rb +5 -2
- data/lib/mutant/config.rb +59 -34
- data/lib/mutant/env.rb +14 -4
- data/lib/mutant/integration.rb +7 -10
- data/lib/mutant/integration/null.rb +0 -1
- data/lib/mutant/isolation.rb +11 -48
- data/lib/mutant/isolation/fork.rb +107 -40
- data/lib/mutant/isolation/none.rb +18 -5
- data/lib/mutant/license/subscription/commercial.rb +2 -3
- data/lib/mutant/license/subscription/opensource.rb +0 -1
- data/lib/mutant/matcher/method/instance.rb +0 -2
- data/lib/mutant/mutator/node/send.rb +1 -1
- data/lib/mutant/parallel.rb +0 -1
- data/lib/mutant/parallel/worker.rb +0 -2
- data/lib/mutant/reporter/cli.rb +0 -2
- data/lib/mutant/reporter/cli/printer/config.rb +9 -5
- data/lib/mutant/reporter/cli/printer/coverage_result.rb +19 -0
- data/lib/mutant/reporter/cli/printer/env_progress.rb +2 -0
- data/lib/mutant/reporter/cli/printer/isolation_result.rb +19 -35
- data/lib/mutant/reporter/cli/printer/mutation_result.rb +4 -9
- data/lib/mutant/reporter/cli/printer/subject_result.rb +2 -2
- data/lib/mutant/result.rb +81 -30
- data/lib/mutant/runner/sink.rb +12 -5
- data/lib/mutant/timer.rb +60 -11
- data/lib/mutant/transform.rb +25 -21
- data/lib/mutant/version.rb +1 -1
- data/lib/mutant/warnings.rb +0 -1
- data/lib/mutant/world.rb +15 -0
- metadata +3 -5
- data/lib/mutant/reporter/cli/printer/mutation_progress_result.rb +0 -28
- data/lib/mutant/reporter/cli/printer/subject_progress.rb +0 -58
- data/lib/mutant/reporter/cli/printer/test_result.rb +0 -32
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: defe26105634d28a81b10e876713ab4e29a53a5d78ac8fe7e2d8a44d32b644c6
         | 
| 4 | 
            +
              data.tar.gz: fad9645931798ed165a249e8e8d4a1c032896ceba6b823a1464ff2f60a7cadbb
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: 79da9f75f18510bbf81e43e2b98e43c4b92ddf2e6cb68b66828f9f4240f7d3e8747d6e475130f83a2d1edec8c1dda43a34f48524bd7439372e545a046099c28d
         | 
| 7 | 
            +
              data.tar.gz: 7d8aac2776eca84aff051b7a146863bb271c5df4ec7dc344f041d3a22a17b4d87f9f48d5201e2dceec30f548adc7135d443ee78bc00764a9399fb677f2cf0aa5
         | 
    
        data/lib/mutant.rb
    CHANGED
    
    | @@ -179,16 +179,14 @@ require 'mutant/reporter/sequence' | |
| 179 179 | 
             
            require 'mutant/reporter/cli'
         | 
| 180 180 | 
             
            require 'mutant/reporter/cli/printer'
         | 
| 181 181 | 
             
            require 'mutant/reporter/cli/printer/config'
         | 
| 182 | 
            +
            require 'mutant/reporter/cli/printer/coverage_result'
         | 
| 182 183 | 
             
            require 'mutant/reporter/cli/printer/env'
         | 
| 183 184 | 
             
            require 'mutant/reporter/cli/printer/env_progress'
         | 
| 184 185 | 
             
            require 'mutant/reporter/cli/printer/env_result'
         | 
| 185 186 | 
             
            require 'mutant/reporter/cli/printer/isolation_result'
         | 
| 186 | 
            -
            require 'mutant/reporter/cli/printer/mutation_progress_result'
         | 
| 187 187 | 
             
            require 'mutant/reporter/cli/printer/mutation_result'
         | 
| 188 188 | 
             
            require 'mutant/reporter/cli/printer/status_progressive'
         | 
| 189 | 
            -
            require 'mutant/reporter/cli/printer/subject_progress'
         | 
| 190 189 | 
             
            require 'mutant/reporter/cli/printer/subject_result'
         | 
| 191 | 
            -
            require 'mutant/reporter/cli/printer/test_result'
         | 
| 192 190 | 
             
            require 'mutant/reporter/cli/format'
         | 
| 193 191 | 
             
            require 'mutant/repository'
         | 
| 194 192 | 
             
            require 'mutant/repository/diff'
         | 
| @@ -219,12 +217,14 @@ module Mutant | |
| 219 217 | 
             
                stderr:             $stderr,
         | 
| 220 218 | 
             
                stdout:             $stdout,
         | 
| 221 219 | 
             
                thread:             Thread,
         | 
| 220 | 
            +
                timer:              Timer.new(Process),
         | 
| 222 221 | 
             
                warnings:           Warnings.new(Warning)
         | 
| 223 222 | 
             
              )
         | 
| 224 223 |  | 
| 225 224 | 
             
              # Reopen class to initialize constant to avoid dep circle
         | 
| 226 225 | 
             
              class Config
         | 
| 227 226 | 
             
                DEFAULT = new(
         | 
| 227 | 
            +
                  coverage_criteria: Config::CoverageCriteria::DEFAULT,
         | 
| 228 228 | 
             
                  expression_parser: Expression::Parser.new([
         | 
| 229 229 | 
             
                    Expression::Method,
         | 
| 230 230 | 
             
                    Expression::Methods,
         | 
| @@ -237,6 +237,7 @@ module Mutant | |
| 237 237 | 
             
                  isolation:         Mutant::Isolation::Fork.new(WORLD),
         | 
| 238 238 | 
             
                  jobs:              nil,
         | 
| 239 239 | 
             
                  matcher:           Matcher::Config::DEFAULT,
         | 
| 240 | 
            +
                  mutation_timeout:  nil,
         | 
| 240 241 | 
             
                  reporter:          Reporter::CLI.build(WORLD.stdout),
         | 
| 241 242 | 
             
                  requires:          EMPTY_ARRAY,
         | 
| 242 243 | 
             
                  zombie:            false
         | 
| @@ -35,7 +35,7 @@ module Mutant | |
| 35 35 |  | 
| 36 36 | 
             
                    def initialize(attributes)
         | 
| 37 37 | 
             
                      super(attributes)
         | 
| 38 | 
            -
                      @config = Config | 
| 38 | 
            +
                      @config = Config.env
         | 
| 39 39 | 
             
                    end
         | 
| 40 40 |  | 
| 41 41 | 
             
                    def execute
         | 
| @@ -49,7 +49,7 @@ module Mutant | |
| 49 49 | 
             
                    end
         | 
| 50 50 |  | 
| 51 51 | 
             
                    def expand(file_config)
         | 
| 52 | 
            -
                      @config =  | 
| 52 | 
            +
                      @config = @config.merge(file_config)
         | 
| 53 53 | 
             
                    end
         | 
| 54 54 |  | 
| 55 55 | 
             
                    def soft_fail(result)
         | 
| @@ -153,6 +153,9 @@ module Mutant | |
| 153 153 | 
             
                      parser.on('-j', '--jobs NUMBER', 'Number of kill jobs. Defaults to number of processors.') do |number|
         | 
| 154 154 | 
             
                        set(jobs: Integer(number))
         | 
| 155 155 | 
             
                      end
         | 
| 156 | 
            +
                      parser.on('-t', '--mutation-timeout NUMBER', 'Per mutation analysis timeout') do |number|
         | 
| 157 | 
            +
                        set(mutation_timeout: Float(number))
         | 
| 158 | 
            +
                      end
         | 
| 156 159 | 
             
                    end
         | 
| 157 160 | 
             
                  end # Run
         | 
| 158 161 | 
             
                  # rubocop:enable Metrics/ClassLength
         | 
    
        data/lib/mutant/config.rb
    CHANGED
    
    | @@ -7,6 +7,7 @@ module Mutant | |
| 7 7 | 
             
              # to current environment is being represented by the Mutant::Env object.
         | 
| 8 8 | 
             
              class Config
         | 
| 9 9 | 
             
                include Adamantium::Flat, Anima.new(
         | 
| 10 | 
            +
                  :coverage_criteria,
         | 
| 10 11 | 
             
                  :expression_parser,
         | 
| 11 12 | 
             
                  :fail_fast,
         | 
| 12 13 | 
             
                  :includes,
         | 
| @@ -14,6 +15,7 @@ module Mutant | |
| 14 15 | 
             
                  :isolation,
         | 
| 15 16 | 
             
                  :jobs,
         | 
| 16 17 | 
             
                  :matcher,
         | 
| 18 | 
            +
                  :mutation_timeout,
         | 
| 17 19 | 
             
                  :reporter,
         | 
| 18 20 | 
             
                  :requires,
         | 
| 19 21 | 
             
                  :zombie
         | 
| @@ -23,30 +25,6 @@ module Mutant | |
| 23 25 | 
             
                  define_method(:"#{name}?") { public_send(name) }
         | 
| 24 26 | 
             
                end
         | 
| 25 27 |  | 
| 26 | 
            -
                boolean = Transform::Boolean.new
         | 
| 27 | 
            -
                integer = Transform::Primitive.new(Integer)
         | 
| 28 | 
            -
                string  = Transform::Primitive.new(String)
         | 
| 29 | 
            -
             | 
| 30 | 
            -
                string_array = Transform::Array.new(string)
         | 
| 31 | 
            -
             | 
| 32 | 
            -
                TRANSFORM = Transform::Sequence.new(
         | 
| 33 | 
            -
                  [
         | 
| 34 | 
            -
                    Transform::Exception.new(SystemCallError, :read.to_proc),
         | 
| 35 | 
            -
                    Transform::Exception.new(YAML::SyntaxError, YAML.method(:safe_load)),
         | 
| 36 | 
            -
                    Transform::Hash.new(
         | 
| 37 | 
            -
                      optional: [
         | 
| 38 | 
            -
                        Transform::Hash::Key.new('fail_fast',   boolean),
         | 
| 39 | 
            -
                        Transform::Hash::Key.new('includes',    string_array),
         | 
| 40 | 
            -
                        Transform::Hash::Key.new('integration', string),
         | 
| 41 | 
            -
                        Transform::Hash::Key.new('jobs',        integer),
         | 
| 42 | 
            -
                        Transform::Hash::Key.new('requires',    string_array)
         | 
| 43 | 
            -
                      ],
         | 
| 44 | 
            -
                      required: []
         | 
| 45 | 
            -
                    ),
         | 
| 46 | 
            -
                    Transform::Hash::Symbolize.new
         | 
| 47 | 
            -
                  ]
         | 
| 48 | 
            -
                )
         | 
| 49 | 
            -
             | 
| 50 28 | 
             
                MORE_THAN_ONE_CONFIG_FILE = <<~'MESSAGE'
         | 
| 51 29 | 
             
                  Found more than one candidate for use as implicit config file: %s
         | 
| 52 30 | 
             
                MESSAGE
         | 
| @@ -57,6 +35,32 @@ module Mutant | |
| 57 35 | 
             
                  mutant.yml
         | 
| 58 36 | 
             
                ].freeze
         | 
| 59 37 |  | 
| 38 | 
            +
                private_constant(*constants(false))
         | 
| 39 | 
            +
             | 
| 40 | 
            +
                class CoverageCriteria
         | 
| 41 | 
            +
                  include Anima.new(:timeout, :test_result)
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                  DEFAULT = new(
         | 
| 44 | 
            +
                    timeout:     false,
         | 
| 45 | 
            +
                    test_result: true
         | 
| 46 | 
            +
                  )
         | 
| 47 | 
            +
             | 
| 48 | 
            +
                  TRANSFORM =
         | 
| 49 | 
            +
                    Transform::Sequence.new(
         | 
| 50 | 
            +
                      [
         | 
| 51 | 
            +
                        Transform::Hash.new(
         | 
| 52 | 
            +
                          optional: [
         | 
| 53 | 
            +
                            Transform::Hash::Key.new('timeout',     Transform::BOOLEAN),
         | 
| 54 | 
            +
                            Transform::Hash::Key.new('test_result', Transform::BOOLEAN)
         | 
| 55 | 
            +
                          ],
         | 
| 56 | 
            +
                          required: []
         | 
| 57 | 
            +
                        ),
         | 
| 58 | 
            +
                        Transform::Hash::Symbolize.new,
         | 
| 59 | 
            +
                        ->(value) { Either::Right.new(DEFAULT.with(**value)) }
         | 
| 60 | 
            +
                      ]
         | 
| 61 | 
            +
                    )
         | 
| 62 | 
            +
                end # CoverageCriteria
         | 
| 63 | 
            +
             | 
| 60 64 | 
             
                # Merge with other config
         | 
| 61 65 | 
             
                #
         | 
| 62 66 | 
             
                # @param [Config] other
         | 
| @@ -64,18 +68,17 @@ module Mutant | |
| 64 68 | 
             
                # @return [Config]
         | 
| 65 69 | 
             
                def merge(other)
         | 
| 66 70 | 
             
                  other.with(
         | 
| 67 | 
            -
                    fail_fast: | 
| 68 | 
            -
                    includes: | 
| 69 | 
            -
                    jobs: | 
| 70 | 
            -
                    integration: | 
| 71 | 
            -
                     | 
| 72 | 
            -
                     | 
| 73 | 
            -
                     | 
| 71 | 
            +
                    fail_fast:        fail_fast || other.fail_fast,
         | 
| 72 | 
            +
                    includes:         other.includes + includes,
         | 
| 73 | 
            +
                    jobs:             other.jobs || jobs,
         | 
| 74 | 
            +
                    integration:      other.integration || integration,
         | 
| 75 | 
            +
                    mutation_timeout: other.mutation_timeout || mutation_timeout,
         | 
| 76 | 
            +
                    matcher:          matcher.merge(other.matcher),
         | 
| 77 | 
            +
                    requires:         other.requires + requires,
         | 
| 78 | 
            +
                    zombie:           zombie || other.zombie
         | 
| 74 79 | 
             
                  )
         | 
| 75 80 | 
             
                end
         | 
| 76 81 |  | 
| 77 | 
            -
                private_constant(*constants(false))
         | 
| 78 | 
            -
             | 
| 79 82 | 
             
                # Load config file
         | 
| 80 83 | 
             
                #
         | 
| 81 84 | 
             
                # @param [World] world
         | 
| @@ -98,7 +101,7 @@ module Mutant | |
| 98 101 | 
             
                def self.load_contents(path)
         | 
| 99 102 | 
             
                  Transform::Named
         | 
| 100 103 | 
             
                    .new(path.to_s, TRANSFORM)
         | 
| 101 | 
            -
                    . | 
| 104 | 
            +
                    .call(path)
         | 
| 102 105 | 
             
                    .lmap(&:compact_message)
         | 
| 103 106 | 
             
                end
         | 
| 104 107 | 
             
                private_class_method :load_contents
         | 
| @@ -109,5 +112,27 @@ module Mutant | |
| 109 112 | 
             
                def self.env
         | 
| 110 113 | 
             
                  DEFAULT.with(jobs: Etc.nprocessors)
         | 
| 111 114 | 
             
                end
         | 
| 115 | 
            +
             | 
| 116 | 
            +
                TRANSFORM = Transform::Sequence.new(
         | 
| 117 | 
            +
                  [
         | 
| 118 | 
            +
                    Transform::Exception.new(SystemCallError, :read.to_proc),
         | 
| 119 | 
            +
                    Transform::Exception.new(YAML::SyntaxError, YAML.method(:safe_load)),
         | 
| 120 | 
            +
                    Transform::Hash.new(
         | 
| 121 | 
            +
                      optional: [
         | 
| 122 | 
            +
                        Transform::Hash::Key.new('coverage_criteria', CoverageCriteria::TRANSFORM),
         | 
| 123 | 
            +
                        Transform::Hash::Key.new('fail_fast',         Transform::BOOLEAN),
         | 
| 124 | 
            +
                        Transform::Hash::Key.new('includes',          Transform::STRING_ARRAY),
         | 
| 125 | 
            +
                        Transform::Hash::Key.new('integration',       Transform::STRING),
         | 
| 126 | 
            +
                        Transform::Hash::Key.new('jobs',              Transform::INTEGER),
         | 
| 127 | 
            +
                        Transform::Hash::Key.new('mutation_timeout',  Transform::FLOAT),
         | 
| 128 | 
            +
                        Transform::Hash::Key.new('requires',          Transform::STRING_ARRAY)
         | 
| 129 | 
            +
                      ],
         | 
| 130 | 
            +
                      required: []
         | 
| 131 | 
            +
                    ),
         | 
| 132 | 
            +
                    Transform::Hash::Symbolize.new
         | 
| 133 | 
            +
                  ]
         | 
| 134 | 
            +
                )
         | 
| 135 | 
            +
             | 
| 136 | 
            +
                private_constant(:TRANSFORM)
         | 
| 112 137 | 
             
              end # Config
         | 
| 113 138 | 
             
            end # Mutant
         | 
    
        data/lib/mutant/env.rb
    CHANGED
    
    | @@ -24,10 +24,15 @@ module Mutant | |
| 24 24 | 
             
                # @param [Config] config
         | 
| 25 25 | 
             
                #
         | 
| 26 26 | 
             
                # @return [Env]
         | 
| 27 | 
            +
                #
         | 
| 28 | 
            +
                # rubocop:disable Metrics/MethodLength
         | 
| 27 29 | 
             
                def self.empty(world, config)
         | 
| 28 30 | 
             
                  new(
         | 
| 29 31 | 
             
                    config:           config,
         | 
| 30 | 
            -
                    integration:      Integration::Null.new( | 
| 32 | 
            +
                    integration:      Integration::Null.new(
         | 
| 33 | 
            +
                      expression_parser: config.expression_parser,
         | 
| 34 | 
            +
                      timer:             world.timer
         | 
| 35 | 
            +
                    ),
         | 
| 31 36 | 
             
                    matchable_scopes: EMPTY_ARRAY,
         | 
| 32 37 | 
             
                    mutations:        EMPTY_ARRAY,
         | 
| 33 38 | 
             
                    parser:           Parser.new,
         | 
| @@ -36,6 +41,7 @@ module Mutant | |
| 36 41 | 
             
                    world:            world
         | 
| 37 42 | 
             
                  )
         | 
| 38 43 | 
             
                end
         | 
| 44 | 
            +
                # rubocop:enable Metrics/MethodLength
         | 
| 39 45 |  | 
| 40 46 | 
             
                # Kill mutation
         | 
| 41 47 | 
             
                #
         | 
| @@ -43,14 +49,14 @@ module Mutant | |
| 43 49 | 
             
                #
         | 
| 44 50 | 
             
                # @return [Result::Mutation]
         | 
| 45 51 | 
             
                def kill(mutation)
         | 
| 46 | 
            -
                  start =  | 
| 52 | 
            +
                  start = timer.now
         | 
| 47 53 |  | 
| 48 54 | 
             
                  tests = selections.fetch(mutation.subject)
         | 
| 49 55 |  | 
| 50 56 | 
             
                  Result::Mutation.new(
         | 
| 51 57 | 
             
                    isolation_result: run_mutation_tests(mutation, tests),
         | 
| 52 58 | 
             
                    mutation:         mutation,
         | 
| 53 | 
            -
                    runtime:           | 
| 59 | 
            +
                    runtime:          timer.now - start
         | 
| 54 60 | 
             
                  )
         | 
| 55 61 | 
             
                end
         | 
| 56 62 |  | 
| @@ -127,7 +133,7 @@ module Mutant | |
| 127 133 | 
             
              private
         | 
| 128 134 |  | 
| 129 135 | 
             
                def run_mutation_tests(mutation, tests)
         | 
| 130 | 
            -
                  config.isolation.call do
         | 
| 136 | 
            +
                  config.isolation.call(config.mutation_timeout) do
         | 
| 131 137 | 
             
                    result = mutation.insert(world.kernel)
         | 
| 132 138 |  | 
| 133 139 | 
             
                    if result.equal?(Loader::Result::VoidValue.instance)
         | 
| @@ -138,5 +144,9 @@ module Mutant | |
| 138 144 | 
             
                  end
         | 
| 139 145 | 
             
                end
         | 
| 140 146 |  | 
| 147 | 
            +
                def timer
         | 
| 148 | 
            +
                  world.timer
         | 
| 149 | 
            +
                end
         | 
| 150 | 
            +
             | 
| 141 151 | 
             
              end # Env
         | 
| 142 152 | 
             
            end # Mutant
         | 
    
        data/lib/mutant/integration.rb
    CHANGED
    
    | @@ -4,7 +4,7 @@ module Mutant | |
| 4 4 |  | 
| 5 5 | 
             
              # Abstract base class mutant test framework integrations
         | 
| 6 6 | 
             
              class Integration
         | 
| 7 | 
            -
                include AbstractType, Adamantium::Flat,  | 
| 7 | 
            +
                include AbstractType, Adamantium::Flat, Anima.new(:expression_parser, :timer)
         | 
| 8 8 |  | 
| 9 9 | 
             
                LOAD_MESSAGE = <<~'MESSAGE'
         | 
| 10 10 | 
             
                  Unable to load integration mutant-%<integration_name>s:
         | 
| @@ -27,9 +27,12 @@ module Mutant | |
| 27 27 | 
             
                #
         | 
| 28 28 | 
             
                # @return [Either<String, Integration>]
         | 
| 29 29 | 
             
                def self.setup(env)
         | 
| 30 | 
            -
                  attempt_require(env)
         | 
| 31 | 
            -
                    . | 
| 32 | 
            -
             | 
| 30 | 
            +
                  attempt_require(env).bind { attempt_const_get(env) }.fmap do |klass|
         | 
| 31 | 
            +
                    klass.new(
         | 
| 32 | 
            +
                      expression_parser: env.config.expression_parser,
         | 
| 33 | 
            +
                      timer:             env.world.timer
         | 
| 34 | 
            +
                    ).setup
         | 
| 35 | 
            +
                  end
         | 
| 33 36 | 
             
                end
         | 
| 34 37 |  | 
| 35 38 | 
             
                # rubocop:disable Style/MultilineBlockChain
         | 
| @@ -80,11 +83,5 @@ module Mutant | |
| 80 83 | 
             
                #
         | 
| 81 84 | 
             
                # @return [Enumerable<Test>]
         | 
| 82 85 | 
             
                abstract_method :all_tests
         | 
| 83 | 
            -
             | 
| 84 | 
            -
              private
         | 
| 85 | 
            -
             | 
| 86 | 
            -
                def expression_parser
         | 
| 87 | 
            -
                  config.expression_parser
         | 
| 88 | 
            -
                end
         | 
| 89 86 | 
             
              end # Integration
         | 
| 90 87 | 
             
            end # Mutant
         | 
    
        data/lib/mutant/isolation.rb
    CHANGED
    
    | @@ -7,57 +7,20 @@ module Mutant | |
| 7 7 |  | 
| 8 8 | 
             
                # Isolated computation result
         | 
| 9 9 | 
             
                class Result
         | 
| 10 | 
            -
                  include  | 
| 11 | 
            -
             | 
| 12 | 
            -
             | 
| 13 | 
            -
             | 
| 14 | 
            -
             | 
| 15 | 
            -
             | 
| 16 | 
            -
                   | 
| 17 | 
            -
             | 
| 18 | 
            -
                   | 
| 19 | 
            -
             | 
| 20 | 
            -
                  # Add error on top of current result
         | 
| 21 | 
            -
                  #
         | 
| 22 | 
            -
                  # @param [Result] error
         | 
| 23 | 
            -
                  #
         | 
| 24 | 
            -
                  # @return [Result]
         | 
| 25 | 
            -
                  def add_error(error)
         | 
| 26 | 
            -
                    ErrorChain.new(error, self)
         | 
| 27 | 
            -
                  end
         | 
| 28 | 
            -
             | 
| 29 | 
            -
                  # The log captured from integration
         | 
| 30 | 
            -
                  #
         | 
| 31 | 
            -
                  # @return [String]
         | 
| 32 | 
            -
                  def log
         | 
| 33 | 
            -
                    NULL_LOG
         | 
| 34 | 
            -
                  end
         | 
| 35 | 
            -
             | 
| 36 | 
            -
                  # Test for success
         | 
| 10 | 
            +
                  include Anima.new(
         | 
| 11 | 
            +
                    :exception,
         | 
| 12 | 
            +
                    :log,
         | 
| 13 | 
            +
                    :process_status,
         | 
| 14 | 
            +
                    :timeout,
         | 
| 15 | 
            +
                    :value
         | 
| 16 | 
            +
                  )
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                  # Test for successful result
         | 
| 37 19 | 
             
                  #
         | 
| 38 20 | 
             
                  # @return [Boolean]
         | 
| 39 | 
            -
                  def  | 
| 40 | 
            -
                     | 
| 21 | 
            +
                  def valid_value?
         | 
| 22 | 
            +
                    timeout.nil? && exception.nil? && (process_status.nil? || process_status.success?)
         | 
| 41 23 | 
             
                  end
         | 
| 42 | 
            -
             | 
| 43 | 
            -
                  # Successful result producing value
         | 
| 44 | 
            -
                  class Success < self
         | 
| 45 | 
            -
                    include Concord::Public.new(:value, :log)
         | 
| 46 | 
            -
             | 
| 47 | 
            -
                    def self.new(_value, _log = '')
         | 
| 48 | 
            -
                      super
         | 
| 49 | 
            -
                    end
         | 
| 50 | 
            -
                  end # Success
         | 
| 51 | 
            -
             | 
| 52 | 
            -
                  # Unsuccessful result by unexpected exception
         | 
| 53 | 
            -
                  class Exception < self
         | 
| 54 | 
            -
                    include Concord::Public.new(:value)
         | 
| 55 | 
            -
                  end # Error
         | 
| 56 | 
            -
             | 
| 57 | 
            -
                  # Result when there where many results
         | 
| 58 | 
            -
                  class ErrorChain < Result
         | 
| 59 | 
            -
                    include Concord::Public.new(:value, :next)
         | 
| 60 | 
            -
                  end # ChainError
         | 
| 61 24 | 
             
                end # Result
         | 
| 62 25 |  | 
| 63 26 | 
             
                # Call block in isolation
         | 
| @@ -3,22 +3,36 @@ | |
| 3 3 | 
             
            module Mutant
         | 
| 4 4 | 
             
              class Isolation
         | 
| 5 5 | 
             
                # Isolation via the fork(2) systemcall.
         | 
| 6 | 
            +
                #
         | 
| 7 | 
            +
                # Communication between parent and child process is done
         | 
| 8 | 
            +
                # via anonymous pipes.
         | 
| 9 | 
            +
                #
         | 
| 10 | 
            +
                # Timeouts are initially handled relatively efficiently via IO.select
         | 
| 11 | 
            +
                # but once the child process pipes are on eof via busy looping on
         | 
| 12 | 
            +
                # waitpid2 with Process::WNOHANG set.
         | 
| 13 | 
            +
                #
         | 
| 14 | 
            +
                # Handling timeouts this way is not the conceptually most
         | 
| 15 | 
            +
                # efficient solution. But its cross platform.
         | 
| 16 | 
            +
                #
         | 
| 17 | 
            +
                # Design constraints:
         | 
| 18 | 
            +
                #
         | 
| 19 | 
            +
                # * Support Linux
         | 
| 20 | 
            +
                # * Support MacOSX
         | 
| 21 | 
            +
                # * Avoid platform specific APIs and code.
         | 
| 22 | 
            +
                # * Only use ruby corelib.
         | 
| 23 | 
            +
                # * Do not use any named resource.
         | 
| 24 | 
            +
                # * Never block on latency inducing systemcall without a
         | 
| 25 | 
            +
                #   timeout.
         | 
| 26 | 
            +
                # * Child process freezing before closing the pipes needs to
         | 
| 27 | 
            +
                #   be detected by parent process.
         | 
| 28 | 
            +
                # * Child process freezing after closing the pipes needs to be
         | 
| 29 | 
            +
                #   detected by parent process.
         | 
| 6 30 | 
             
                class Fork < self
         | 
| 7 31 | 
             
                  include(Adamantium::Flat, Concord.new(:world))
         | 
| 8 32 |  | 
| 9 33 | 
             
                  READ_SIZE = 4096
         | 
| 10 34 |  | 
| 11 | 
            -
                  ATTRIBUTES = %i[block log_pipe result_pipe world].freeze
         | 
| 12 | 
            -
             | 
| 13 | 
            -
                  # Unsuccessful result as child exited nonzero
         | 
| 14 | 
            -
                  class ChildError < Result
         | 
| 15 | 
            -
                    include Concord::Public.new(:value, :log)
         | 
| 16 | 
            -
                  end # ChildError
         | 
| 17 | 
            -
             | 
| 18 | 
            -
                  # Unsuccessful result as fork failed
         | 
| 19 | 
            -
                  class ForkError < Result
         | 
| 20 | 
            -
                    include Equalizer.new
         | 
| 21 | 
            -
                  end # ForkError
         | 
| 35 | 
            +
                  ATTRIBUTES = %i[block deadline log_pipe result_pipe world].freeze
         | 
| 22 36 |  | 
| 23 37 | 
             
                  # Pipe abstraction
         | 
| 24 38 | 
             
                  class Pipe
         | 
| @@ -50,7 +64,6 @@ module Mutant | |
| 50 64 | 
             
                    end
         | 
| 51 65 | 
             
                  end # Pipe
         | 
| 52 66 |  | 
| 53 | 
            -
                  # ignore :reek:InstanceVariableAssumption
         | 
| 54 67 | 
             
                  class Parent
         | 
| 55 68 | 
             
                    include(
         | 
| 56 69 | 
             
                      Anima.new(*ATTRIBUTES),
         | 
| @@ -64,15 +77,28 @@ module Mutant | |
| 64 77 | 
             
                    #
         | 
| 65 78 | 
             
                    # @return [Result]
         | 
| 66 79 | 
             
                    def call
         | 
| 67 | 
            -
                       | 
| 68 | 
            -
             | 
| 69 | 
            -
                       | 
| 70 | 
            -
             | 
| 71 | 
            -
                      @ | 
| 80 | 
            +
                      @exception     = nil
         | 
| 81 | 
            +
                      @log_fragments = []
         | 
| 82 | 
            +
                      @timeout       = nil
         | 
| 83 | 
            +
                      @value         = nil
         | 
| 84 | 
            +
                      @pid           = start_child
         | 
| 85 | 
            +
             | 
| 86 | 
            +
                      read_child_result
         | 
| 87 | 
            +
                      result
         | 
| 72 88 | 
             
                    end
         | 
| 73 89 |  | 
| 74 90 | 
             
                  private
         | 
| 75 91 |  | 
| 92 | 
            +
                    def result
         | 
| 93 | 
            +
                      Result.new(
         | 
| 94 | 
            +
                        exception:      @exception,
         | 
| 95 | 
            +
                        log:            @log_fragments.join,
         | 
| 96 | 
            +
                        process_status: @process_status,
         | 
| 97 | 
            +
                        timeout:        @timeout,
         | 
| 98 | 
            +
                        value:          @value
         | 
| 99 | 
            +
                      )
         | 
| 100 | 
            +
                    end
         | 
| 101 | 
            +
             | 
| 76 102 | 
             
                    def start_child
         | 
| 77 103 | 
             
                      world.process.fork do
         | 
| 78 104 | 
             
                        Child.call(
         | 
| @@ -85,30 +111,43 @@ module Mutant | |
| 85 111 | 
             
                    end
         | 
| 86 112 |  | 
| 87 113 | 
             
                    # rubocop:disable Metrics/MethodLength
         | 
| 88 | 
            -
                    def read_child_result | 
| 114 | 
            +
                    def read_child_result
         | 
| 89 115 | 
             
                      result_fragments = []
         | 
| 90 | 
            -
                      log_fragments    = []
         | 
| 91 116 |  | 
| 92 | 
            -
                       | 
| 93 | 
            -
                         | 
| 94 | 
            -
             | 
| 95 | 
            -
             | 
| 117 | 
            +
                      targets =
         | 
| 118 | 
            +
                        {
         | 
| 119 | 
            +
                          log_pipe.parent    => @log_fragments,
         | 
| 120 | 
            +
                          result_pipe.parent => result_fragments
         | 
| 121 | 
            +
                        }
         | 
| 96 122 |  | 
| 97 | 
            -
                       | 
| 98 | 
            -
             | 
| 99 | 
            -
                       | 
| 100 | 
            -
                         | 
| 123 | 
            +
                      read_targets(targets)
         | 
| 124 | 
            +
             | 
| 125 | 
            +
                      if targets.empty?
         | 
| 126 | 
            +
                        load_result(result_fragments)
         | 
| 127 | 
            +
                        terminate_graceful
         | 
| 101 128 | 
             
                      else
         | 
| 102 | 
            -
                         | 
| 129 | 
            +
                        @timeout = deadline.allowed_time
         | 
| 130 | 
            +
                        terminate_ungraceful
         | 
| 103 131 | 
             
                      end
         | 
| 104 | 
            -
                    ensure
         | 
| 105 | 
            -
                      wait_child(pid, log_fragments)
         | 
| 106 132 | 
             
                    end
         | 
| 107 133 | 
             
                    # rubocop:enable Metrics/MethodLength
         | 
| 108 134 |  | 
| 109 | 
            -
                    def  | 
| 135 | 
            +
                    def load_result(result_fragments)
         | 
| 136 | 
            +
                      @value = world.marshal.load(result_fragments.join)
         | 
| 137 | 
            +
                    rescue ArgumentError => exception
         | 
| 138 | 
            +
                      @exception = exception
         | 
| 139 | 
            +
                    end
         | 
| 140 | 
            +
             | 
| 141 | 
            +
                    # rubocop:disable Metrics/MethodLength
         | 
| 142 | 
            +
                    def read_targets(targets)
         | 
| 110 143 | 
             
                      until targets.empty?
         | 
| 111 | 
            -
                         | 
| 144 | 
            +
                        status = deadline.status
         | 
| 145 | 
            +
             | 
| 146 | 
            +
                        break unless status.ok?
         | 
| 147 | 
            +
             | 
| 148 | 
            +
                        ready, = world.io.select(targets.keys, [], [], status.time_left)
         | 
| 149 | 
            +
             | 
| 150 | 
            +
                        break unless ready
         | 
| 112 151 |  | 
| 113 152 | 
             
                        ready.each do |fd|
         | 
| 114 153 | 
             
                          if fd.eof?
         | 
| @@ -119,14 +158,42 @@ module Mutant | |
| 119 158 | 
             
                        end
         | 
| 120 159 | 
             
                      end
         | 
| 121 160 | 
             
                    end
         | 
| 161 | 
            +
                    # rubocop:enable Metrics/MethodLength
         | 
| 122 162 |  | 
| 123 | 
            -
                     | 
| 124 | 
            -
             | 
| 163 | 
            +
                    # rubocop:disable Metrics/MethodLength
         | 
| 164 | 
            +
                    def terminate_graceful
         | 
| 165 | 
            +
                      status = nil
         | 
| 166 | 
            +
             | 
| 167 | 
            +
                      loop do
         | 
| 168 | 
            +
                        status = peek_child
         | 
| 169 | 
            +
                        break if status || deadline.expired?
         | 
| 170 | 
            +
                        world.kernel.sleep(0.1)
         | 
| 171 | 
            +
                      end
         | 
| 125 172 |  | 
| 126 | 
            -
                       | 
| 127 | 
            -
                         | 
| 173 | 
            +
                      if status
         | 
| 174 | 
            +
                        handle_status(status)
         | 
| 175 | 
            +
                      else
         | 
| 176 | 
            +
                        terminate_ungraceful
         | 
| 128 177 | 
             
                      end
         | 
| 129 178 | 
             
                    end
         | 
| 179 | 
            +
                    # rubocop:enable Metrics/MethodLength
         | 
| 180 | 
            +
             | 
| 181 | 
            +
                    def terminate_ungraceful
         | 
| 182 | 
            +
                      world.process.kill('KILL', @pid)
         | 
| 183 | 
            +
             | 
| 184 | 
            +
                      _pid, status = world.process.wait2(@pid)
         | 
| 185 | 
            +
             | 
| 186 | 
            +
                      handle_status(status)
         | 
| 187 | 
            +
                    end
         | 
| 188 | 
            +
             | 
| 189 | 
            +
                    def handle_status(status)
         | 
| 190 | 
            +
                      @process_status = status
         | 
| 191 | 
            +
                    end
         | 
| 192 | 
            +
             | 
| 193 | 
            +
                    def peek_child
         | 
| 194 | 
            +
                      _pid, status = world.process.wait2(@pid, Process::WNOHANG)
         | 
| 195 | 
            +
                      status
         | 
| 196 | 
            +
                    end
         | 
| 130 197 |  | 
| 131 198 | 
             
                    def add_result(result)
         | 
| 132 199 | 
             
                      @result = defined?(@result) ? @result.add_error(result) : result
         | 
| @@ -157,17 +224,16 @@ module Mutant | |
| 157 224 | 
             
                  # Call block in isolation
         | 
| 158 225 | 
             
                  #
         | 
| 159 226 | 
             
                  # @return [Result]
         | 
| 160 | 
            -
                  #   execution result
         | 
| 161 | 
            -
                  #
         | 
| 162 | 
            -
                  # ignore :reek:NestedIterators
         | 
| 163 227 | 
             
                  #
         | 
| 164 228 | 
             
                  # rubocop:disable Metrics/MethodLength
         | 
| 165 | 
            -
                  def call(&block)
         | 
| 229 | 
            +
                  def call(timeout, &block)
         | 
| 230 | 
            +
                    deadline = world.deadline(timeout)
         | 
| 166 231 | 
             
                    io = world.io
         | 
| 167 232 | 
             
                    Pipe.with(io) do |result|
         | 
| 168 233 | 
             
                      Pipe.with(io) do |log|
         | 
| 169 234 | 
             
                        Parent.call(
         | 
| 170 235 | 
             
                          block:       block,
         | 
| 236 | 
            +
                          deadline:    deadline,
         | 
| 171 237 | 
             
                          log_pipe:    log,
         | 
| 172 238 | 
             
                          result_pipe: result,
         | 
| 173 239 | 
             
                          world:       world
         | 
| @@ -176,6 +242,7 @@ module Mutant | |
| 176 242 | 
             
                    end
         | 
| 177 243 | 
             
                  end
         | 
| 178 244 | 
             
                  # rubocop:enable Metrics/MethodLength
         | 
| 245 | 
            +
             | 
| 179 246 | 
             
                end # Fork
         | 
| 180 247 | 
             
              end # Isolation
         | 
| 181 248 | 
             
            end # Mutant
         |