knapsack_pro 8.0.2 → 8.1.0

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 6135451be036cbd75b78cb5efd31498b91b08698771c78c69dfa7b2b1d218faa
4
- data.tar.gz: bd73f2525bc9b89c5faca60bf95249d06e525fb533422b8e98399b03dcf732bf
3
+ metadata.gz: 9392d6bfe6377c70af5278f4b527314194c5516f5693badc4255d46c989de2c6
4
+ data.tar.gz: 559425cde606e2dc159912dd3facf07467c5fa1ac507fced6734c0b655f246bf
5
5
  SHA512:
6
- metadata.gz: 3cca977135a8795f725cd2ee73ae5c8af2df313db76892a9e29e283fc788f0c7479dff1e8aed532fb43a1ac56cb4582200dda0af1e953426c571fb177eb2ad2f
7
- data.tar.gz: 00bb2a047303b2e11eb06f1f0f358dd209a1d98cb01657b4c8144a7f137efc5cb7aa44e736e6f4925642975afb83b576c2e9a0039233b389a196180ca0a194d3
6
+ metadata.gz: 35a9edadc691088d01e48fdde8f496d9ed1bebb49d9dff50ffe9887c0238a5bf4b5c4289fcfcc024afd9921b53a8c5a28500ae2a00d0ca860f6ad2bfdae81de4
7
+ data.tar.gz: edeea08e31dfe17e0d614922665f93e76e0adbbf8564e56ef0d1f751bb38355e9e3b6f2bbd98d4f2d0ffb6a3ad417318497a150087723cf193b2f4e01f215e9f
data/CHANGELOG.md CHANGED
@@ -1,6 +1,16 @@
1
1
  # Changelog
2
2
 
3
- ### Unreleased
3
+ ### 8.1.0
4
+
5
+ * Improve the performance of the [RSpec split by test examples](https://docs.knapsackpro.com/ruby/split-by-test-examples/).
6
+
7
+ https://github.com/KnapsackPro/knapsack_pro-ruby/pull/292
8
+
9
+ * Reduce `/v1/build_distributions/last` API requests.
10
+ * (Queue Mode) Improve the speed of starting tests for CI nodes that are started after the queue was already initialized by another CI node.
11
+ * (Regular Mode) Improve the speed of starting tests for CI nodes that are started after the test suite split was already initialized by another CI node.
12
+
13
+ https://github.com/KnapsackPro/knapsack_pro-ruby/compare/v8.0.2...v8.1.0
4
14
 
5
15
  ### 8.0.2
6
16
 
@@ -23,28 +23,8 @@ module KnapsackPro
23
23
  KnapsackPro::Config::Env.test_dir || TestFilePattern.test_dir(adapter_class)
24
24
  end
25
25
 
26
- # In Fallback Mode, we always want to run whole test files (not split by
27
- # test cases) to guarantee that each test will be executed at least once
28
- # across parallel CI nodes.
29
- def fallback_mode_test_files
30
- all_test_files_to_run
31
- end
32
-
33
- # Detect test files present on the disk that should be run.
34
- # This may include fast test files + slow test files split by test cases.
35
- def fast_and_slow_test_files_to_run
36
- test_files_to_run = all_test_files_to_run
37
-
38
- if adapter_class.split_by_test_cases_enabled?
39
- slow_test_files = get_slow_test_files
40
- return test_files_to_run if slow_test_files.empty?
41
-
42
- test_file_cases = adapter_class.test_file_cases_for(slow_test_files)
43
-
44
- return KnapsackPro::TestFilesWithTestCasesComposer.call(test_files_to_run, slow_test_files, test_file_cases)
45
- end
46
-
47
- test_files_to_run
26
+ def test_suite
27
+ KnapsackPro::TestSuite.new(adapter_class)
48
28
  end
49
29
 
50
30
  private
@@ -58,29 +38,5 @@ module KnapsackPro
58
38
  def repository_adapter
59
39
  KnapsackPro::RepositoryAdapterInitiator.call
60
40
  end
61
-
62
- def test_file_pattern
63
- TestFilePattern.call(adapter_class)
64
- end
65
-
66
- def all_test_files_to_run
67
- KnapsackPro::TestFileFinder.call(test_file_pattern)
68
- end
69
-
70
- def slow_test_file_pattern
71
- KnapsackPro::Config::Env.slow_test_file_pattern
72
- end
73
-
74
- def get_slow_test_files
75
- slow_test_files =
76
- if slow_test_file_pattern
77
- KnapsackPro::TestFileFinder.slow_test_files_by_pattern(adapter_class)
78
- else
79
- # get slow test files from API and ensure they exist on disk
80
- KnapsackPro::SlowTestFileFinder.call(adapter_class)
81
- end
82
- KnapsackPro.logger.debug("Detected #{slow_test_files.size} slow test files: #{slow_test_files.inspect}")
83
- slow_test_files
84
- end
85
41
  end
86
42
  end
@@ -5,13 +5,10 @@ module KnapsackPro
5
5
  class Decryptor
6
6
  class TooManyEncryptedTestFilesError < StandardError; end
7
7
 
8
- def self.call(test_files, encrypted_test_files)
9
- if KnapsackPro::Config::Env.test_files_encrypted?
10
- new(test_files, encrypted_test_files).call
11
- else
12
- # those test files are not encrypted
13
- encrypted_test_files
14
- end
8
+ def self.call(test_suite, test_files)
9
+ return test_files unless KnapsackPro::Config::Env.test_files_encrypted?
10
+
11
+ new(test_suite.calculate_test_files.test_files, test_files).call
15
12
  end
16
13
 
17
14
  def initialize(test_files, encrypted_test_files)
@@ -4,72 +4,87 @@ module KnapsackPro
4
4
  class QueueAllocator
5
5
  FallbackModeError = Class.new(StandardError)
6
6
 
7
+ class Batch
8
+ def initialize(connection, response)
9
+ @connection = connection
10
+ @response = response
11
+
12
+ raise ArgumentError.new(connection.response) if connection.errors?
13
+ end
14
+
15
+ def queue_exists?
16
+ raise "Connection failed. Please report this as a bug: #{KnapsackPro::Urls::SUPPORT}" if connection_failed?
17
+ return false if connection.api_code == KnapsackPro::Client::API::V1::Queues::CODE_ATTEMPT_CONNECT_TO_QUEUE_FAILED
18
+
19
+ true
20
+ end
21
+
22
+ def connection_failed?
23
+ !connection.success?
24
+ end
25
+
26
+ def test_files
27
+ response.fetch('test_files')
28
+ end
29
+
30
+ private
31
+
32
+ attr_reader :connection, :response
33
+ end
34
+
7
35
  def initialize(args)
8
- @fast_and_slow_test_files_to_run = args.fetch(:fast_and_slow_test_files_to_run)
9
- @fallback_mode_test_files = args.fetch(:fallback_mode_test_files)
36
+ @test_suite = args.fetch(:test_suite)
10
37
  @ci_node_total = args.fetch(:ci_node_total)
11
38
  @ci_node_index = args.fetch(:ci_node_index)
12
39
  @ci_node_build_id = args.fetch(:ci_node_build_id)
13
40
  @repository_adapter = args.fetch(:repository_adapter)
41
+ @fallback_mode = false
14
42
  end
15
43
 
16
44
  def test_file_paths(can_initialize_queue, executed_test_files)
17
- return [] if @fallback_activated
18
- action = build_action(can_initialize_queue, attempt_connect_to_queue: can_initialize_queue)
19
- connection = KnapsackPro::Client::Connection.new(action)
20
- response = connection.call
45
+ return [] if @fallback_mode
21
46
 
22
- # when attempt to connect to existing queue on API side failed because queue does not exist yet
23
- if can_initialize_queue && connection.success? && connection.api_code == KnapsackPro::Client::API::V1::Queues::CODE_ATTEMPT_CONNECT_TO_QUEUE_FAILED
24
- # make attempt to initalize a new queue on API side
25
- action = build_action(can_initialize_queue, attempt_connect_to_queue: false)
26
- connection = KnapsackPro::Client::Connection.new(action)
27
- response = connection.call
28
- end
47
+ batch = pull_tests_from_queue(can_initialize_queue)
29
48
 
30
- if connection.success?
31
- raise ArgumentError.new(response) if connection.errors?
32
- prepare_test_files(response)
33
- elsif !KnapsackPro::Config::Env.fallback_mode_enabled?
34
- message = "Fallback Mode was disabled with KNAPSACK_PRO_FALLBACK_MODE_ENABLED=false. Please restart this CI node to retry tests. Most likely Fallback Mode was disabled due to #{KnapsackPro::Urls::QUEUE_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_FALSE}"
35
- KnapsackPro.logger.error(message)
36
- raise FallbackModeError.new(message)
37
- elsif KnapsackPro::Config::Env.ci_node_retry_count > 0
38
- message = "knapsack_pro gem could not connect to Knapsack Pro API and the Fallback Mode cannot be used this time. Running tests in Fallback Mode are not allowed for retried parallel CI node to avoid running the wrong set of tests. Please manually retry this parallel job on your CI server then knapsack_pro gem will try to connect to Knapsack Pro API again and will run a correct set of tests for this CI node. Learn more #{KnapsackPro::Urls::QUEUE_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_TRUE_AND_POSITIVE_RETRY_COUNT}"
39
- unless KnapsackPro::Config::Env.fixed_queue_split?
40
- message += " Please ensure you have set KNAPSACK_PRO_FIXED_QUEUE_SPLIT=true to allow Knapsack Pro API remember the recorded CI node tests so when you retry failed tests on the CI node then the same set of tests will be executed. See more #{KnapsackPro::Urls::FIXED_QUEUE_SPLIT}"
41
- end
42
- KnapsackPro.logger.error(message)
43
- raise FallbackModeError.new(message)
44
- else
45
- @fallback_activated = true
46
- KnapsackPro.logger.warn("Fallback mode started. We could not connect with Knapsack Pro API. Your tests will be executed based on directory names. If other CI nodes were able to connect with Knapsack Pro API then you may notice that some of the test files will be executed twice across CI nodes. The most important thing is to guarantee each of test files is run at least once! Read more about fallback mode at #{KnapsackPro::Urls::FALLBACK_MODE}")
47
- fallback_test_files(executed_test_files)
48
- end
49
+ return switch_to_fallback_mode(executed_test_files: executed_test_files) if batch.connection_failed?
50
+ return normalize_test_files(batch.test_files) if batch.queue_exists?
51
+
52
+ test_files_result = test_suite.calculate_test_files
53
+
54
+ return try_initializing_queue(test_files_result.test_files) if test_files_result.quick?
55
+
56
+ # The tests to run were found slowly. By that time, the queue could have already been initialized by another CI node.
57
+ # Attempt to pull tests from the queue to avoid the attempt to initialize the queue unnecessarily (queue initialization is an expensive request with a big test files payload).
58
+ batch = pull_tests_from_queue(can_initialize_queue)
59
+
60
+ return switch_to_fallback_mode(executed_test_files: executed_test_files) if batch.connection_failed?
61
+ return normalize_test_files(batch.test_files) if batch.queue_exists?
62
+
63
+ try_initializing_queue(test_files_result.test_files)
49
64
  end
50
65
 
51
66
  private
52
67
 
53
- attr_reader :fast_and_slow_test_files_to_run,
54
- :fallback_mode_test_files,
68
+ attr_reader :test_suite,
55
69
  :ci_node_total,
56
70
  :ci_node_index,
57
71
  :ci_node_build_id,
58
72
  :repository_adapter
59
73
 
60
- def encrypted_test_files
61
- KnapsackPro::Crypto::Encryptor.call(fast_and_slow_test_files_to_run)
62
- end
63
-
64
74
  def encrypted_branch
65
75
  KnapsackPro::Crypto::BranchEncryptor.call(repository_adapter.branch)
66
76
  end
67
77
 
68
- def build_action(can_initialize_queue, attempt_connect_to_queue:)
69
- test_files =
70
- if can_initialize_queue && !attempt_connect_to_queue
71
- encrypted_test_files
72
- end
78
+ def normalize_test_files(test_files)
79
+ decrypted_test_files = KnapsackPro::Crypto::Decryptor.call(test_suite, test_files)
80
+ KnapsackPro::TestFilePresenter.paths(decrypted_test_files)
81
+ end
82
+
83
+ def build_action(can_initialize_queue:, attempt_connect_to_queue:, test_files: nil)
84
+ if can_initialize_queue && !attempt_connect_to_queue
85
+ raise 'Test files are required when initializing a new queue.' if test_files.nil?
86
+ test_files = KnapsackPro::Crypto::Encryptor.call(test_files)
87
+ end
73
88
 
74
89
  KnapsackPro::Client::API::V1::Queues.queue(
75
90
  can_initialize_queue: can_initialize_queue,
@@ -83,13 +98,50 @@ module KnapsackPro
83
98
  )
84
99
  end
85
100
 
86
- def prepare_test_files(response)
87
- decrypted_test_files = KnapsackPro::Crypto::Decryptor.call(fast_and_slow_test_files_to_run, response['test_files'])
88
- KnapsackPro::TestFilePresenter.paths(decrypted_test_files)
101
+ def pull_tests_from_queue(can_initialize_queue)
102
+ action = build_action(can_initialize_queue: can_initialize_queue, attempt_connect_to_queue: can_initialize_queue)
103
+ connection = KnapsackPro::Client::Connection.new(action)
104
+ response = connection.call
105
+ Batch.new(connection, response)
106
+ end
107
+
108
+ def initialize_queue(tests_to_run)
109
+ action = build_action(can_initialize_queue: true, attempt_connect_to_queue: false, test_files: tests_to_run)
110
+ connection = KnapsackPro::Client::Connection.new(action)
111
+ response = connection.call
112
+ Batch.new(connection, response)
113
+ end
114
+
115
+ def try_initializing_queue(tests)
116
+ result = initialize_queue(tests)
117
+
118
+ return switch_to_fallback_mode(executed_test_files: []) if result.connection_failed?
119
+
120
+ normalize_test_files(result.test_files)
121
+ end
122
+
123
+ def switch_to_fallback_mode(executed_test_files:)
124
+ @fallback_mode = true
125
+
126
+ if !KnapsackPro::Config::Env.fallback_mode_enabled?
127
+ message = "Fallback Mode was disabled with KNAPSACK_PRO_FALLBACK_MODE_ENABLED=false. Please restart this CI node to retry tests. Most likely Fallback Mode was disabled due to #{KnapsackPro::Urls::QUEUE_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_FALSE}"
128
+ KnapsackPro.logger.error(message)
129
+ raise FallbackModeError.new(message)
130
+ elsif KnapsackPro::Config::Env.ci_node_retry_count > 0
131
+ message = "knapsack_pro gem could not connect to Knapsack Pro API and the Fallback Mode cannot be used this time. Running tests in Fallback Mode are not allowed for retried parallel CI node to avoid running the wrong set of tests. Please manually retry this parallel job on your CI server then knapsack_pro gem will try to connect to Knapsack Pro API again and will run a correct set of tests for this CI node. Learn more #{KnapsackPro::Urls::QUEUE_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_TRUE_AND_POSITIVE_RETRY_COUNT}"
132
+ unless KnapsackPro::Config::Env.fixed_queue_split?
133
+ message += " Please ensure you have set KNAPSACK_PRO_FIXED_QUEUE_SPLIT=true to allow Knapsack Pro API remember the recorded CI node tests so when you retry failed tests on the CI node then the same set of tests will be executed. See more #{KnapsackPro::Urls::FIXED_QUEUE_SPLIT}"
134
+ end
135
+ KnapsackPro.logger.error(message)
136
+ raise FallbackModeError.new(message)
137
+ else
138
+ KnapsackPro.logger.warn("Fallback mode started. We could not connect with Knapsack Pro API. Your tests will be executed based on directory names. If other CI nodes were able to connect with Knapsack Pro API then you may notice that some of the test files will be executed twice across CI nodes. The most important thing is to guarantee each of test files is run at least once! Read more about fallback mode at #{KnapsackPro::Urls::FALLBACK_MODE}")
139
+ fallback_test_files(executed_test_files)
140
+ end
89
141
  end
90
142
 
91
143
  def fallback_test_files(executed_test_files)
92
- test_flat_distributor = KnapsackPro::TestFlatDistributor.new(fallback_mode_test_files, ci_node_total)
144
+ test_flat_distributor = KnapsackPro::TestFlatDistributor.new(test_suite.fallback_test_files, ci_node_total)
93
145
  test_files_for_node_index = test_flat_distributor.test_files_for_node(ci_node_index)
94
146
  KnapsackPro::TestFilePresenter.paths(test_files_for_node_index) - executed_test_files
95
147
  end
@@ -4,8 +4,7 @@ module KnapsackPro
4
4
  class QueueAllocatorBuilder < BaseAllocatorBuilder
5
5
  def allocator
6
6
  KnapsackPro::QueueAllocator.new(
7
- fast_and_slow_test_files_to_run: fast_and_slow_test_files_to_run,
8
- fallback_mode_test_files: fallback_mode_test_files,
7
+ test_suite: test_suite,
9
8
  ci_node_total: env.ci_node_total,
10
9
  ci_node_index: env.ci_node_index,
11
10
  ci_node_build_id: env.ci_node_build_id,
@@ -1,71 +1,83 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module KnapsackPro
4
- class Allocator
4
+ class RegularAllocator
5
+ class Split
6
+ def initialize(connection, response)
7
+ @connection = connection
8
+ @response = response
9
+
10
+ raise ArgumentError.new(connection.response) if connection.errors?
11
+ end
12
+
13
+ def exists?
14
+ raise "Connection failed. Please report this as a bug: #{KnapsackPro::Urls::SUPPORT}" if connection_failed?
15
+ return false if connection.api_code == KnapsackPro::Client::API::V1::BuildDistributions::TEST_SUITE_SPLIT_CACHE_MISS_CODE
16
+
17
+ true
18
+ end
19
+
20
+ def connection_failed?
21
+ !connection.success?
22
+ end
23
+
24
+ def test_files
25
+ response.fetch('test_files')
26
+ end
27
+
28
+ private
29
+
30
+ attr_reader :connection, :response
31
+ end
32
+
5
33
  def initialize(args)
6
- @fast_and_slow_test_files_to_run = args.fetch(:fast_and_slow_test_files_to_run)
7
- @fallback_mode_test_files = args.fetch(:fallback_mode_test_files)
34
+ @test_suite = args.fetch(:test_suite)
8
35
  @ci_node_total = args.fetch(:ci_node_total)
9
36
  @ci_node_index = args.fetch(:ci_node_index)
10
37
  @repository_adapter = args.fetch(:repository_adapter)
11
38
  end
12
39
 
13
40
  def test_file_paths
14
- action = build_action(cache_read_attempt: true)
15
- connection = KnapsackPro::Client::Connection.new(action)
16
- response = connection.call
41
+ split = pull_tests
17
42
 
18
- # when a cache miss because the test suite split was not cached yet
19
- if connection.success? && connection.api_code == KnapsackPro::Client::API::V1::BuildDistributions::TEST_SUITE_SPLIT_CACHE_MISS_CODE
20
- # make an attempt to initalize a new test suite split on the API side
21
- action = build_action(cache_read_attempt: false)
22
- connection = KnapsackPro::Client::Connection.new(action)
23
- response = connection.call
24
- end
43
+ return switch_to_fallback_mode if split.connection_failed?
44
+ return normalize_test_files(split.test_files) if split.exists?
25
45
 
26
- if connection.success?
27
- raise ArgumentError.new(response) if connection.errors?
28
- prepare_test_files(response)
29
- elsif !KnapsackPro::Config::Env.fallback_mode_enabled?
30
- message = "Fallback Mode was disabled with KNAPSACK_PRO_FALLBACK_MODE_ENABLED=false. Please restart this CI node to retry tests. Most likely Fallback Mode was disabled due to #{KnapsackPro::Urls::REGULAR_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_FALSE}"
31
- KnapsackPro.logger.error(message)
32
- exit_code = KnapsackPro::Config::Env.fallback_mode_error_exit_code
33
- Kernel.exit(exit_code)
34
- elsif KnapsackPro::Config::Env.ci_node_retry_count > 0
35
- message = "knapsack_pro gem could not connect to Knapsack Pro API and the Fallback Mode cannot be used this time. Running tests in Fallback Mode are not allowed for retried parallel CI node to avoid running the wrong set of tests. Please manually retry this parallel job on your CI server then knapsack_pro gem will try to connect to Knapsack Pro API again and will run a correct set of tests for this CI node. Learn more #{KnapsackPro::Urls::REGULAR_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_TRUE_AND_POSITIVE_RETRY_COUNT}"
36
- unless KnapsackPro::Config::Env.fixed_test_suite_split?
37
- message += " Please ensure you have set KNAPSACK_PRO_FIXED_TEST_SUITE_SPLIT=true to allow Knapsack Pro API remember the recorded CI node tests so when you retry failed tests on the CI node then the same set of tests will be executed. See more #{KnapsackPro::Urls::FIXED_TEST_SUITE_SPLIT}"
38
- end
39
- KnapsackPro.logger.error(message)
40
- exit_code = KnapsackPro::Config::Env.fallback_mode_error_exit_code
41
- Kernel.exit(exit_code)
42
- else
43
- KnapsackPro.logger.warn("Fallback mode started. We could not connect with Knapsack Pro API. Your tests will be executed based on directory names. Read more about fallback mode at #{KnapsackPro::Urls::FALLBACK_MODE}")
44
- fallback_test_files
45
- end
46
+ test_files_result = test_suite.calculate_test_files
47
+
48
+ return try_initializing_test_suite_split(test_files_result.test_files) if test_files_result.quick?
49
+
50
+ # The tests to run were found slowly. By that time, the test suite split could have already been initialized by another CI node.
51
+ # Attempt to pull tests to avoid the attempt to initialize the test suite split unnecessarily (test suite split initialization is an expensive request with a big test files payload).
52
+ split = pull_tests
53
+
54
+ return switch_to_fallback_mode if split.connection_failed?
55
+ return normalize_test_files(split.test_files) if split.exists?
56
+
57
+ try_initializing_test_suite_split(test_files_result.test_files)
46
58
  end
47
59
 
48
60
  private
49
61
 
50
- attr_reader :fast_and_slow_test_files_to_run,
51
- :fallback_mode_test_files,
62
+ attr_reader :test_suite,
52
63
  :ci_node_total,
53
64
  :ci_node_index,
54
65
  :repository_adapter
55
66
 
56
- def encrypted_test_files
57
- KnapsackPro::Crypto::Encryptor.call(fast_and_slow_test_files_to_run)
58
- end
59
-
60
67
  def encrypted_branch
61
68
  KnapsackPro::Crypto::BranchEncryptor.call(repository_adapter.branch)
62
69
  end
63
70
 
64
- def build_action(cache_read_attempt:)
65
- test_files =
66
- unless cache_read_attempt
67
- encrypted_test_files
68
- end
71
+ def normalize_test_files(test_files)
72
+ decrypted_test_files = KnapsackPro::Crypto::Decryptor.call(test_suite, test_files)
73
+ KnapsackPro::TestFilePresenter.paths(decrypted_test_files)
74
+ end
75
+
76
+ def build_action(cache_read_attempt:, test_files: nil)
77
+ unless cache_read_attempt
78
+ raise 'Test files are required when initializing a new test suite split.' if test_files.nil?
79
+ test_files = KnapsackPro::Crypto::Encryptor.call(test_files)
80
+ end
69
81
 
70
82
  KnapsackPro::Client::API::V1::BuildDistributions.subset(
71
83
  cache_read_attempt: cache_read_attempt,
@@ -77,13 +89,50 @@ module KnapsackPro
77
89
  )
78
90
  end
79
91
 
80
- def prepare_test_files(response)
81
- decrypted_test_files = KnapsackPro::Crypto::Decryptor.call(fast_and_slow_test_files_to_run, response['test_files'])
82
- KnapsackPro::TestFilePresenter.paths(decrypted_test_files)
92
+ def pull_tests
93
+ action = build_action(cache_read_attempt: true)
94
+ connection = KnapsackPro::Client::Connection.new(action)
95
+ response = connection.call
96
+ Split.new(connection, response)
97
+ end
98
+
99
+ def initialize_test_suite_split(tests_to_run)
100
+ action = build_action(cache_read_attempt: false, test_files: tests_to_run)
101
+ connection = KnapsackPro::Client::Connection.new(action)
102
+ response = connection.call
103
+ Split.new(connection, response)
104
+ end
105
+
106
+ def try_initializing_test_suite_split(tests)
107
+ split = initialize_test_suite_split(tests)
108
+
109
+ return switch_to_fallback_mode if split.connection_failed?
110
+
111
+ normalize_test_files(split.test_files)
112
+ end
113
+
114
+ def switch_to_fallback_mode
115
+ if !KnapsackPro::Config::Env.fallback_mode_enabled?
116
+ message = "Fallback Mode was disabled with KNAPSACK_PRO_FALLBACK_MODE_ENABLED=false. Please restart this CI node to retry tests. Most likely Fallback Mode was disabled due to #{KnapsackPro::Urls::REGULAR_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_FALSE}"
117
+ KnapsackPro.logger.error(message)
118
+ exit_code = KnapsackPro::Config::Env.fallback_mode_error_exit_code
119
+ Kernel.exit(exit_code)
120
+ elsif KnapsackPro::Config::Env.ci_node_retry_count > 0
121
+ message = "knapsack_pro gem could not connect to Knapsack Pro API and the Fallback Mode cannot be used this time. Running tests in Fallback Mode are not allowed for retried parallel CI node to avoid running the wrong set of tests. Please manually retry this parallel job on your CI server then knapsack_pro gem will try to connect to Knapsack Pro API again and will run a correct set of tests for this CI node. Learn more #{KnapsackPro::Urls::REGULAR_MODE__CONNECTION_ERROR_WITH_FALLBACK_ENABLED_TRUE_AND_POSITIVE_RETRY_COUNT}"
122
+ unless KnapsackPro::Config::Env.fixed_test_suite_split?
123
+ message += " Please ensure you have set KNAPSACK_PRO_FIXED_TEST_SUITE_SPLIT=true to allow Knapsack Pro API remember the recorded CI node tests so when you retry failed tests on the CI node then the same set of tests will be executed. See more #{KnapsackPro::Urls::FIXED_TEST_SUITE_SPLIT}"
124
+ end
125
+ KnapsackPro.logger.error(message)
126
+ exit_code = KnapsackPro::Config::Env.fallback_mode_error_exit_code
127
+ Kernel.exit(exit_code)
128
+ else
129
+ KnapsackPro.logger.warn("Fallback mode started. We could not connect with Knapsack Pro API. Your tests will be executed based on directory names. Read more about fallback mode at #{KnapsackPro::Urls::FALLBACK_MODE}")
130
+ fallback_test_files
131
+ end
83
132
  end
84
133
 
85
134
  def fallback_test_files
86
- test_flat_distributor = KnapsackPro::TestFlatDistributor.new(fallback_mode_test_files, ci_node_total)
135
+ test_flat_distributor = KnapsackPro::TestFlatDistributor.new(test_suite.fallback_test_files, ci_node_total)
87
136
  test_files_for_node_index = test_flat_distributor.test_files_for_node(ci_node_index)
88
137
  KnapsackPro::TestFilePresenter.paths(test_files_for_node_index)
89
138
  end
@@ -1,11 +1,10 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module KnapsackPro
4
- class AllocatorBuilder < BaseAllocatorBuilder
4
+ class RegularAllocatorBuilder < BaseAllocatorBuilder
5
5
  def allocator
6
- KnapsackPro::Allocator.new(
7
- fast_and_slow_test_files_to_run: fast_and_slow_test_files_to_run,
8
- fallback_mode_test_files: fallback_mode_test_files,
6
+ KnapsackPro::RegularAllocator.new(
7
+ test_suite: test_suite,
9
8
  ci_node_total: env.ci_node_total,
10
9
  ci_node_index: env.ci_node_index,
11
10
  repository_adapter: repository_adapter,
@@ -8,7 +8,7 @@ module KnapsackPro
8
8
  end
9
9
 
10
10
  def initialize(adapter_class)
11
- @allocator_builder = KnapsackPro::AllocatorBuilder.new(adapter_class)
11
+ @allocator_builder = KnapsackPro::RegularAllocatorBuilder.new(adapter_class)
12
12
  @allocator = allocator_builder.allocator
13
13
  end
14
14
 
@@ -0,0 +1,63 @@
1
+ # frozen_string_literal: true
2
+
3
+ module KnapsackPro
4
+ class TestSuite
5
+ Result = Struct.new(:test_files, :quick?)
6
+
7
+ def initialize(adapter_class)
8
+ @adapter_class = adapter_class
9
+ end
10
+
11
+ # Detect test files present on the disk that should be run.
12
+ # This may include fast test files + slow test files split by test cases.
13
+ def calculate_test_files
14
+ return @result if defined?(@result)
15
+
16
+ unless adapter_class.split_by_test_cases_enabled?
17
+ return @result = Result.new(all_test_files_to_run, true)
18
+ end
19
+
20
+ slow_test_files, quick =
21
+ if slow_test_file_pattern
22
+ [KnapsackPro::TestFileFinder.slow_test_files_by_pattern(adapter_class), true]
23
+ else
24
+ [KnapsackPro::SlowTestFileFinder.call(adapter_class), false]
25
+ end
26
+
27
+ KnapsackPro.logger.debug("Detected #{slow_test_files.size} slow test files: #{slow_test_files.inspect}")
28
+
29
+ if slow_test_files.empty?
30
+ return @result = Result.new(all_test_files_to_run, quick)
31
+ end
32
+
33
+ test_file_cases = adapter_class.test_file_cases_for(slow_test_files)
34
+
35
+ fast_files_and_cases_for_slow_tests = KnapsackPro::TestFilesWithTestCasesComposer.call(all_test_files_to_run, slow_test_files, test_file_cases)
36
+
37
+ @result = Result.new(fast_files_and_cases_for_slow_tests, false)
38
+ end
39
+
40
+ # In Fallback Mode, we always want to run whole test files (not split by
41
+ # test cases) to guarantee that each test will be executed at least once
42
+ # across parallel CI nodes.
43
+ def fallback_test_files
44
+ all_test_files_to_run
45
+ end
46
+
47
+ private
48
+
49
+ attr_reader :adapter_class
50
+
51
+ def all_test_files_to_run
52
+ @all_test_files_to_run ||= KnapsackPro::TestFileFinder.call(test_file_pattern)
53
+ end
54
+
55
+ def test_file_pattern
56
+ TestFilePattern.call(adapter_class)
57
+ end
58
+
59
+ def slow_test_file_pattern
60
+ KnapsackPro::Config::Env.slow_test_file_pattern
61
+ end
62
+ end
63
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module KnapsackPro
4
- VERSION = '8.0.2'
4
+ VERSION = '8.1.0'
5
5
  end
data/lib/knapsack_pro.rb CHANGED
@@ -55,9 +55,10 @@ require_relative 'knapsack_pro/adapters/cucumber_adapter'
55
55
  require_relative 'knapsack_pro/adapters/minitest_adapter'
56
56
  require_relative 'knapsack_pro/adapters/test_unit_adapter'
57
57
  require_relative 'knapsack_pro/adapters/spinach_adapter'
58
- require_relative 'knapsack_pro/allocator'
58
+ require_relative 'knapsack_pro/regular_allocator'
59
59
  require_relative 'knapsack_pro/queue_allocator'
60
60
  require_relative 'knapsack_pro/mask_string'
61
+ require_relative 'knapsack_pro/test_suite'
61
62
  require_relative 'knapsack_pro/test_case_mergers/base_merger'
62
63
  require_relative 'knapsack_pro/test_case_mergers/rspec_merger'
63
64
  require_relative 'knapsack_pro/build_distribution_fetcher'
@@ -65,7 +66,7 @@ require_relative 'knapsack_pro/slow_test_file_determiner'
65
66
  require_relative 'knapsack_pro/slow_test_file_finder'
66
67
  require_relative 'knapsack_pro/test_files_with_test_cases_composer'
67
68
  require_relative 'knapsack_pro/base_allocator_builder'
68
- require_relative 'knapsack_pro/allocator_builder'
69
+ require_relative 'knapsack_pro/regular_allocator_builder'
69
70
  require_relative 'knapsack_pro/queue_allocator_builder'
70
71
  require_relative 'knapsack_pro/batch'
71
72
  require_relative 'knapsack_pro/queue'
@@ -2022,8 +2022,6 @@ describe "#{KnapsackPro::Runners::Queue::RSpecRunner} - Integration tests", :cle
2022
2022
 
2023
2023
  actual = subject
2024
2024
 
2025
- expect(actual.stdout).to match(/DEBUG -- : \[knapsack_pro\] Detected 1 slow test files: \[\{"path".?=>.?"spec_integration\/a_spec\.rb"\}\]/)
2026
-
2027
2025
  expect(actual.stdout).to include(
2028
2026
  <<~OUTPUT
2029
2027
  A_describe
@@ -2474,8 +2472,6 @@ describe "#{KnapsackPro::Runners::Queue::RSpecRunner} - Integration tests", :cle
2474
2472
 
2475
2473
  actual = subject
2476
2474
 
2477
- expect(actual.stdout).to include('DEBUG -- : [knapsack_pro] Detected 0 slow test files: []')
2478
-
2479
2475
  expect(actual.stdout).to include(
2480
2476
  <<~OUTPUT
2481
2477
  A_describe