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 +4 -4
- data/CHANGELOG.md +11 -1
- data/lib/knapsack_pro/base_allocator_builder.rb +2 -46
- data/lib/knapsack_pro/crypto/decryptor.rb +4 -7
- data/lib/knapsack_pro/queue_allocator.rb +99 -47
- data/lib/knapsack_pro/queue_allocator_builder.rb +1 -2
- data/lib/knapsack_pro/{allocator.rb → regular_allocator.rb} +97 -48
- data/lib/knapsack_pro/{allocator_builder.rb → regular_allocator_builder.rb} +3 -4
- data/lib/knapsack_pro/runners/base_runner.rb +1 -1
- data/lib/knapsack_pro/test_suite.rb +63 -0
- data/lib/knapsack_pro/version.rb +1 -1
- data/lib/knapsack_pro.rb +3 -2
- data/spec/integration/runners/queue/rspec_runner_spec.rb +0 -4
- data/spec/knapsack_pro/base_allocator_builder_spec.rb +0 -102
- data/spec/knapsack_pro/crypto/decryptor_spec.rb +3 -1
- data/spec/knapsack_pro/runners/base_runner_spec.rb +3 -3
- metadata +4 -11
- data/spec/knapsack_pro/allocator_builder_spec.rb +0 -36
- data/spec/knapsack_pro/allocator_spec.rb +0 -233
- data/spec/knapsack_pro/queue_allocator_builder_spec.rb +0 -39
- data/spec/knapsack_pro/queue_allocator_spec.rb +0 -304
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 9392d6bfe6377c70af5278f4b527314194c5516f5693badc4255d46c989de2c6
|
4
|
+
data.tar.gz: 559425cde606e2dc159912dd3facf07467c5fa1ac507fced6734c0b655f246bf
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 35a9edadc691088d01e48fdde8f496d9ed1bebb49d9dff50ffe9887c0238a5bf4b5c4289fcfcc024afd9921b53a8c5a28500ae2a00d0ca860f6ad2bfdae81de4
|
7
|
+
data.tar.gz: edeea08e31dfe17e0d614922665f93e76e0adbbf8564e56ef0d1f751bb38355e9e3b6f2bbd98d4f2d0ffb6a3ad417318497a150087723cf193b2f4e01f215e9f
|
data/CHANGELOG.md
CHANGED
@@ -1,6 +1,16 @@
|
|
1
1
|
# Changelog
|
2
2
|
|
3
|
-
###
|
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
|
-
|
27
|
-
|
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(
|
9
|
-
|
10
|
-
|
11
|
-
|
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
|
-
@
|
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 @
|
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
|
-
|
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
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
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 :
|
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
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
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
|
87
|
-
|
88
|
-
KnapsackPro::
|
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(
|
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
|
-
|
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
|
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
|
-
@
|
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
|
-
|
15
|
-
connection = KnapsackPro::Client::Connection.new(action)
|
16
|
-
response = connection.call
|
41
|
+
split = pull_tests
|
17
42
|
|
18
|
-
|
19
|
-
|
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
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
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 :
|
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
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
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
|
81
|
-
|
82
|
-
KnapsackPro::
|
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(
|
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
|
4
|
+
class RegularAllocatorBuilder < BaseAllocatorBuilder
|
5
5
|
def allocator
|
6
|
-
KnapsackPro::
|
7
|
-
|
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::
|
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
|
data/lib/knapsack_pro/version.rb
CHANGED
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/
|
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/
|
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
|