dependabot-python 0.301.1 → 0.303.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.
@@ -1,6 +1,7 @@
1
- # typed: true
1
+ # typed: strict
2
2
  # frozen_string_literal: true
3
3
 
4
+ require "sorbet-runtime"
4
5
  require "toml-rb"
5
6
  require "open3"
6
7
  require "dependabot/dependency"
@@ -18,59 +19,86 @@ module Dependabot
18
19
  class FileUpdater
19
20
  class PoetryFileUpdater
20
21
  require_relative "pyproject_preparer"
22
+ extend T::Sig
21
23
 
22
- attr_reader :dependencies
24
+ sig { returns(T::Array[Dependabot::DependencyFile]) }
23
25
  attr_reader :dependency_files
26
+
27
+ sig { returns(T::Array[Dependabot::Credential]) }
24
28
  attr_reader :credentials
25
29
 
30
+ sig { returns(T::Array[Dependabot::Dependency]) }
31
+ attr_reader :dependencies
32
+
33
+ sig do
34
+ params(
35
+ dependencies: T::Array[Dependabot::Dependency],
36
+ dependency_files: T::Array[Dependabot::DependencyFile],
37
+ credentials: T::Array[Dependabot::Credential]
38
+ ).void
39
+ end
26
40
  def initialize(dependencies:, dependency_files:, credentials:)
27
41
  @dependencies = dependencies
28
42
  @dependency_files = dependency_files
29
43
  @credentials = credentials
30
- end
31
-
44
+ @updated_dependency_files = T.let(nil, T.nilable(T::Array[Dependabot::DependencyFile]))
45
+ @prepared_pyproject = T.let(nil, T.nilable(String))
46
+ @pyproject = T.let(nil, T.nilable(Dependabot::DependencyFile))
47
+ @lockfile = T.let(nil, T.nilable(Dependabot::DependencyFile))
48
+ @updated_lockfile_content = T.let(nil, T.nilable(String))
49
+ @language_version_manager = T.let(nil, T.nilable(LanguageVersionManager))
50
+ @python_requirement_parser = T.let(nil, T.nilable(FileParser::PythonRequirementParser))
51
+ @updated_pyproject_content = T.let(nil, T.nilable(String))
52
+ @python_helper_path = T.let(nil, T.nilable(String))
53
+ @poetry_lock = T.let(nil, T.nilable(Dependabot::DependencyFile))
54
+ end
55
+
56
+ sig { returns(T::Array[Dependabot::DependencyFile]) }
32
57
  def updated_dependency_files
33
58
  @updated_dependency_files ||= fetch_updated_dependency_files
34
59
  end
35
60
 
36
61
  private
37
62
 
63
+ sig { returns(Dependabot::Dependency) }
38
64
  def dependency
39
65
  # For now, we'll only ever be updating a single dependency
40
- dependencies.first
66
+ T.must(dependencies.first)
41
67
  end
42
68
 
69
+ sig { returns(T::Array[Dependabot::DependencyFile]) }
43
70
  def fetch_updated_dependency_files
44
71
  updated_files = []
45
72
 
46
- if file_changed?(pyproject)
73
+ if file_changed?(T.must(pyproject))
47
74
  updated_files <<
48
75
  updated_file(
49
- file: pyproject,
50
- content: updated_pyproject_content
76
+ file: T.must(pyproject),
77
+ content: T.must(updated_pyproject_content)
51
78
  )
52
79
  end
53
80
 
54
- raise "Expected lockfile to change!" if lockfile && lockfile.content == updated_lockfile_content
81
+ raise "Expected lockfile to change!" if lockfile && lockfile&.content == updated_lockfile_content
55
82
 
56
83
  if lockfile
57
84
  updated_files <<
58
- updated_file(file: lockfile, content: updated_lockfile_content)
85
+ updated_file(file: T.must(lockfile), content: updated_lockfile_content)
59
86
  end
60
87
 
61
88
  updated_files
62
89
  end
63
90
 
91
+ sig { returns(T.nilable(String)) }
64
92
  def updated_pyproject_content
65
- content = pyproject.content
66
- return content unless requirement_changed?(pyproject, dependency)
93
+ content = T.must(pyproject).content
94
+ return content unless requirement_changed?(T.must(pyproject), dependency)
67
95
 
68
96
  updated_content = content.dup
69
97
 
70
- dependency.requirements.zip(dependency.previous_requirements).each do |new_r, old_r|
71
- next unless new_r[:file] == pyproject.name && old_r[:file] == pyproject.name
98
+ dependency.requirements.zip(T.must(dependency.previous_requirements)).each do |new_r, old_r|
99
+ next unless new_r[:file] == pyproject&.name && T.must(old_r)[:file] == pyproject&.name
72
100
 
73
- updated_content = replace_dep(dependency, updated_content, new_r, old_r)
101
+ updated_content = replace_dep(dependency, T.must(updated_content), new_r, T.must(old_r))
74
102
  end
75
103
 
76
104
  raise DependencyFileContentNotChanged, "Content did not change!" if content == updated_content
@@ -78,6 +106,14 @@ module Dependabot
78
106
  updated_content
79
107
  end
80
108
 
109
+ sig do
110
+ params(
111
+ dep: Dependabot::Dependency,
112
+ content: String,
113
+ new_r: T::Hash[Symbol, T.untyped],
114
+ old_r: T::Hash[Symbol, T.untyped]
115
+ ).returns(String)
116
+ end
81
117
  def replace_dep(dep, content, new_r, old_r)
82
118
  new_req = new_r[:requirement]
83
119
  old_req = old_r[:requirement]
@@ -86,8 +122,8 @@ module Dependabot
86
122
  declaration_match = content.match(declaration_regex)
87
123
  if declaration_match
88
124
  declaration = declaration_match[:declaration]
89
- new_declaration = declaration.sub(old_req, new_req)
90
- content.sub(declaration, new_declaration)
125
+ new_declaration = T.must(declaration).sub(old_req, new_req)
126
+ content.sub(T.must(declaration), new_declaration)
91
127
  else
92
128
  content.gsub(table_declaration_regex(dep, new_r)) do |match|
93
129
  match.gsub(/(\s*version\s*=\s*["'])#{Regexp.escape(old_req)}/,
@@ -96,12 +132,13 @@ module Dependabot
96
132
  end
97
133
  end
98
134
 
135
+ sig { returns(String) }
99
136
  def updated_lockfile_content
100
137
  @updated_lockfile_content ||=
101
138
  begin
102
139
  new_lockfile = updated_lockfile_content_for(prepared_pyproject)
103
140
 
104
- original_locked_python = TomlRB.parse(lockfile.content)["metadata"]["python-versions"]
141
+ original_locked_python = TomlRB.parse(T.must(lockfile).content)["metadata"]["python-versions"]
105
142
 
106
143
  new_lockfile.gsub!(/\[metadata\]\n.*python-versions[^\n]+\n/m) do |match|
107
144
  match.gsub(/(["']).*(['"])\n\Z/, '\1' + original_locked_python + '\1' + "\n")
@@ -109,17 +146,18 @@ module Dependabot
109
146
 
110
147
  tmp_hash =
111
148
  TomlRB.parse(new_lockfile)["metadata"]["content-hash"]
112
- correct_hash = pyproject_hash_for(updated_pyproject_content)
149
+ correct_hash = pyproject_hash_for(updated_pyproject_content.to_s)
113
150
 
114
- new_lockfile.gsub(tmp_hash, correct_hash)
151
+ new_lockfile.gsub(tmp_hash, T.must(correct_hash).to_s)
115
152
  end
116
153
  end
117
154
 
155
+ sig { returns(String) }
118
156
  def prepared_pyproject
119
157
  @prepared_pyproject ||=
120
158
  begin
121
159
  content = updated_pyproject_content
122
- content = sanitize(content)
160
+ content = sanitize(T.must(content))
123
161
  content = freeze_other_dependencies(content)
124
162
  content = freeze_dependencies_being_updated(content)
125
163
  content = update_python_requirement(content)
@@ -127,18 +165,20 @@ module Dependabot
127
165
  end
128
166
  end
129
167
 
168
+ sig { params(pyproject_content: String).returns(String) }
130
169
  def freeze_other_dependencies(pyproject_content)
131
170
  PyprojectPreparer
132
171
  .new(pyproject_content: pyproject_content, lockfile: lockfile)
133
172
  .freeze_top_level_dependencies_except(dependencies)
134
173
  end
135
174
 
175
+ sig { params(pyproject_content: String).returns(String) }
136
176
  def freeze_dependencies_being_updated(pyproject_content)
137
177
  pyproject_object = TomlRB.parse(pyproject_content)
138
178
  poetry_object = pyproject_object.fetch("tool").fetch("poetry")
139
179
 
140
180
  dependencies.each do |dep|
141
- if dep.requirements.find { |r| r[:file] == pyproject.name }
181
+ if dep.requirements.find { |r| r[:file] == pyproject&.name }
142
182
  lock_declaration_to_new_version!(poetry_object, dep)
143
183
  else
144
184
  create_declaration_at_new_version!(poetry_object, dep)
@@ -148,12 +188,14 @@ module Dependabot
148
188
  TomlRB.dump(pyproject_object)
149
189
  end
150
190
 
191
+ sig { params(pyproject_content: String).returns(String) }
151
192
  def update_python_requirement(pyproject_content)
152
193
  PyprojectPreparer
153
194
  .new(pyproject_content: pyproject_content)
154
195
  .update_python_requirement(language_version_manager.python_version)
155
196
  end
156
197
 
198
+ sig { params(poetry_object: T::Hash[String, T.untyped], dep: Dependabot::Dependency).returns(T::Array[String]) }
157
199
  def lock_declaration_to_new_version!(poetry_object, dep)
158
200
  Dependabot::Python::FileParser::PyprojectFilesParser::POETRY_DEPENDENCY_TYPES.each do |type|
159
201
  names = poetry_object[type]&.keys || []
@@ -168,6 +210,7 @@ module Dependabot
168
210
  end
169
211
  end
170
212
 
213
+ sig { params(poetry_object: T::Hash[String, T.untyped], dep: Dependabot::Dependency).void }
171
214
  def create_declaration_at_new_version!(poetry_object, dep)
172
215
  subdep_type = dep.production? ? "dependencies" : "dev-dependencies"
173
216
 
@@ -175,12 +218,14 @@ module Dependabot
175
218
  poetry_object[subdep_type][dep.name] = dep.version
176
219
  end
177
220
 
221
+ sig { params(pyproject_content: String).returns(String) }
178
222
  def sanitize(pyproject_content)
179
223
  PyprojectPreparer
180
224
  .new(pyproject_content: pyproject_content)
181
225
  .sanitize
182
226
  end
183
227
 
228
+ sig { params(pyproject_content: String).returns(String) }
184
229
  def updated_lockfile_content_for(pyproject_content)
185
230
  SharedHelpers.in_a_temporary_directory do
186
231
  SharedHelpers.with_git_configured(credentials: credentials) do
@@ -201,6 +246,7 @@ module Dependabot
201
246
 
202
247
  # Using `--lock` avoids doing an install.
203
248
  # Using `--no-interaction` avoids asking for passwords.
249
+ sig { returns(String) }
204
250
  def run_poetry_update_command
205
251
  run_poetry_command(
206
252
  "pyenv exec poetry update #{dependency.name} --lock --no-interaction",
@@ -208,10 +254,12 @@ module Dependabot
208
254
  )
209
255
  end
210
256
 
257
+ sig { params(command: String, fingerprint: T.nilable(String)).returns(String) }
211
258
  def run_poetry_command(command, fingerprint: nil)
212
259
  SharedHelpers.run_shell_command(command, fingerprint: fingerprint)
213
260
  end
214
261
 
262
+ sig { params(pyproject_content: Object).returns(Integer) }
215
263
  def write_temporary_dependency_files(pyproject_content)
216
264
  dependency_files.each do |file|
217
265
  path = file.name
@@ -226,12 +274,22 @@ module Dependabot
226
274
  File.write("pyproject.toml", pyproject_content)
227
275
  end
228
276
 
277
+ sig { void }
229
278
  def add_auth_env_vars
230
279
  Python::FileUpdater::PyprojectPreparer
231
- .new(pyproject_content: pyproject.content)
280
+ .new(pyproject_content: T.must(pyproject&.content))
232
281
  .add_auth_env_vars(credentials)
233
282
  end
234
283
 
284
+ sig do
285
+ params(
286
+ pyproject_content: String
287
+ ).returns(T.nilable(T.any(
288
+ T::Hash[String, T.untyped],
289
+ String,
290
+ T::Array[T::Hash[String, T.untyped]]
291
+ )))
292
+ end
235
293
  def pyproject_hash_for(pyproject_content)
236
294
  SharedHelpers.in_a_temporary_directory do |dir|
237
295
  SharedHelpers.with_git_configured(credentials: credentials) do
@@ -246,6 +304,7 @@ module Dependabot
246
304
  end
247
305
  end
248
306
 
307
+ sig { params(dep: Dependabot::Dependency, old_req: T::Hash[Symbol, T.untyped]).returns(Regexp) }
249
308
  def declaration_regex(dep, old_req)
250
309
  group = old_req[:groups].first
251
310
 
@@ -253,35 +312,42 @@ module Dependabot
253
312
  /#{header_regex}\n.*?(?<declaration>(?:^\s*|["'])#{escape(dep)}["']?\s*=[^\n]*)$/mi
254
313
  end
255
314
 
315
+ sig { params(dep: Dependabot::Dependency, old_req: T::Hash[Symbol, T.untyped]).returns(Regexp) }
256
316
  def table_declaration_regex(dep, old_req)
257
317
  /tool\.poetry\.#{old_req[:groups].first}\.#{escape(dep)}\]\n.*?\s*version\s* =.*?\n/m
258
318
  end
259
319
 
320
+ sig { params(dep: Dependency).returns(String) }
260
321
  def escape(dep)
261
322
  Regexp.escape(dep.name).gsub("\\-", "[-_.]")
262
323
  end
263
324
 
325
+ sig { params(file: Dependabot::DependencyFile).returns(T::Boolean) }
264
326
  def file_changed?(file)
265
327
  dependencies.any? { |dep| requirement_changed?(file, dep) }
266
328
  end
267
329
 
330
+ sig { params(file: Dependabot::DependencyFile, dependency: Dependabot::Dependency).returns(T::Boolean) }
268
331
  def requirement_changed?(file, dependency)
269
332
  changed_requirements =
270
- dependency.requirements - dependency.previous_requirements
333
+ dependency.requirements - T.must(dependency.previous_requirements)
271
334
 
272
335
  changed_requirements.any? { |f| f[:file] == file.name }
273
336
  end
274
337
 
338
+ sig { params(file: Dependabot::DependencyFile, content: String).returns(Dependabot::DependencyFile) }
275
339
  def updated_file(file:, content:)
276
340
  updated_file = file.dup
277
341
  updated_file.content = content
278
342
  updated_file
279
343
  end
280
344
 
345
+ sig { params(name: String).returns(String) }
281
346
  def normalise(name)
282
347
  NameNormaliser.normalise(name)
283
348
  end
284
349
 
350
+ sig { returns(FileParser::PythonRequirementParser) }
285
351
  def python_requirement_parser
286
352
  @python_requirement_parser ||=
287
353
  FileParser::PythonRequirementParser.new(
@@ -289,6 +355,7 @@ module Dependabot
289
355
  )
290
356
  end
291
357
 
358
+ sig { returns(Dependabot::Python::LanguageVersionManager) }
292
359
  def language_version_manager
293
360
  @language_version_manager ||=
294
361
  LanguageVersionManager.new(
@@ -296,19 +363,23 @@ module Dependabot
296
363
  )
297
364
  end
298
365
 
366
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
299
367
  def pyproject
300
368
  @pyproject ||=
301
369
  dependency_files.find { |f| f.name == "pyproject.toml" }
302
370
  end
303
371
 
372
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
304
373
  def lockfile
305
374
  @lockfile ||= poetry_lock
306
375
  end
307
376
 
377
+ sig { returns(String) }
308
378
  def python_helper_path
309
379
  NativeHelpers.python_helper_path
310
380
  end
311
381
 
382
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
312
383
  def poetry_lock
313
384
  dependency_files.find { |f| f.name == "poetry.lock" }
314
385
  end
@@ -1,8 +1,8 @@
1
- # typed: true
1
+ # typed: strict
2
2
  # frozen_string_literal: true
3
3
 
4
4
  require "toml-rb"
5
-
5
+ require "sorbet-runtime"
6
6
  require "dependabot/dependency"
7
7
  require "dependabot/python/file_parser"
8
8
  require "dependabot/python/file_updater"
@@ -14,13 +14,18 @@ module Dependabot
14
14
  module Python
15
15
  class FileUpdater
16
16
  class PyprojectPreparer
17
+ extend T::Sig
18
+
19
+ sig { params(pyproject_content: String, lockfile: T.nilable(Dependabot::DependencyFile)).void }
17
20
  def initialize(pyproject_content:, lockfile: nil)
18
21
  @pyproject_content = pyproject_content
19
22
  @lockfile = lockfile
23
+ @parsed_lockfile = T.let(nil, T.nilable(T::Hash[String, T.untyped]))
20
24
  end
21
25
 
22
26
  # For hosted Dependabot token will be nil since the credentials aren't present.
23
27
  # This is for those running Dependabot themselves and for dry-run.
28
+ sig { params(credentials: T.nilable(T::Array[Dependabot::Credential])).void }
24
29
  def add_auth_env_vars(credentials)
25
30
  TomlRB.parse(@pyproject_content).dig("tool", "poetry", "source")&.each do |source|
26
31
  cred = credentials&.find { |c| c["index-url"] == source["url"] }
@@ -37,6 +42,7 @@ module Dependabot
37
42
  end
38
43
  end
39
44
 
45
+ sig { params(requirement: String).returns(String) }
40
46
  def update_python_requirement(requirement)
41
47
  pyproject_object = TomlRB.parse(@pyproject_content)
42
48
  if (python_specification = pyproject_object.dig("tool", "poetry", "dependencies", "python"))
@@ -48,6 +54,7 @@ module Dependabot
48
54
  TomlRB.dump(pyproject_object)
49
55
  end
50
56
 
57
+ sig { returns(String) }
51
58
  def sanitize
52
59
  # {{ name }} syntax not allowed
53
60
  pyproject_content
@@ -57,6 +64,7 @@ module Dependabot
57
64
 
58
65
  # rubocop:disable Metrics/PerceivedComplexity
59
66
  # rubocop:disable Metrics/AbcSize
67
+ sig { params(dependencies: T::Array[Dependabot::Dependency]).returns(String) }
60
68
  def freeze_top_level_dependencies_except(dependencies)
61
69
  return pyproject_content unless lockfile
62
70
 
@@ -75,14 +83,14 @@ module Dependabot
75
83
 
76
84
  next unless (locked_version = locked_details&.fetch("version"))
77
85
 
78
- next if source_types.include?(locked_details&.dig("source", "type"))
86
+ next if source_types.include?(locked_details.dig("source", "type"))
79
87
 
80
- if locked_details&.dig("source", "type") == "git"
88
+ if locked_details.dig("source", "type") == "git"
81
89
  poetry_object[key][dep_name] = {
82
- "git" => locked_details&.dig("source", "url"),
83
- "rev" => locked_details&.dig("source", "reference")
90
+ "git" => locked_details.dig("source", "url"),
91
+ "rev" => locked_details.dig("source", "reference")
84
92
  }
85
- subdirectory = locked_details&.dig("source", "subdirectory")
93
+ subdirectory = locked_details.dig("source", "subdirectory")
86
94
  poetry_object[key][dep_name]["subdirectory"] = subdirectory if subdirectory
87
95
  elsif poetry_object[key][dep_name].is_a?(Hash)
88
96
  poetry_object[key][dep_name]["version"] = locked_version
@@ -103,20 +111,25 @@ module Dependabot
103
111
 
104
112
  private
105
113
 
114
+ sig { returns(String) }
106
115
  attr_reader :pyproject_content
116
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
107
117
  attr_reader :lockfile
108
118
 
119
+ sig { params(dep_name: String).returns(T.nilable(T::Hash[String, T.untyped])) }
109
120
  def locked_details(dep_name)
110
121
  parsed_lockfile.fetch("package")
111
122
  .find { |d| d["name"] == normalise(dep_name) }
112
123
  end
113
124
 
125
+ sig { params(name: String).returns(String) }
114
126
  def normalise(name)
115
127
  NameNormaliser.normalise(name)
116
128
  end
117
129
 
130
+ sig { returns(T::Hash[String, T.untyped]) }
118
131
  def parsed_lockfile
119
- @parsed_lockfile ||= TomlRB.parse(lockfile.content)
132
+ @parsed_lockfile ||= TomlRB.parse(lockfile&.content)
120
133
  end
121
134
  end
122
135
  end
@@ -114,12 +114,12 @@ module Dependabot
114
114
 
115
115
  sig { returns(T::Array[DependencyFile]) }
116
116
  def updated_pip_compile_based_files
117
- PipCompileFileUpdater.new(
117
+ T.must(PipCompileFileUpdater.new(
118
118
  dependencies: dependencies,
119
119
  dependency_files: dependency_files,
120
120
  credentials: credentials,
121
121
  index_urls: pip_compile_index_urls
122
- ).updated_dependency_files
122
+ ).updated_dependency_files)
123
123
  end
124
124
 
125
125
  sig { returns(T::Array[DependencyFile]) }
@@ -11,28 +11,43 @@ module Dependabot
11
11
 
12
12
  class Language < Dependabot::Ecosystem::VersionManager
13
13
  extend T::Sig
14
- # These versions should match the versions specified at the top of `python/Dockerfile`
15
- PYTHON_3_13 = "3.13"
16
- PYTHON_3_12 = "3.12"
17
- PYTHON_3_11 = "3.11"
18
- PYTHON_3_10 = "3.10"
19
- PYTHON_3_9 = "3.9"
20
- PYTHON_3_8 = "3.8"
14
+ # This list must match the versions specified at the top of `python/Dockerfile`
15
+ # ARG PY_3_13=3.13.2
16
+ PRE_INSTALLED_PYTHON_VERSIONS_RAW = %w(
17
+ 3.13.2
18
+ 3.12.9
19
+ 3.11.11
20
+ 3.10.16
21
+ 3.9.21
22
+ ).freeze
21
23
 
22
- DEPRECATED_VERSIONS = T.let([Version.new(PYTHON_3_8)].freeze, T::Array[Dependabot::Version])
24
+ PRE_INSTALLED_PYTHON_VERSIONS = T.let(PRE_INSTALLED_PYTHON_VERSIONS_RAW.map do |v|
25
+ Version.new(v)
26
+ end.sort, T::Array[Dependabot::Python::Version])
23
27
 
24
- # Keep versions in ascending order
25
- SUPPORTED_VERSIONS = T.let([
26
- Version.new(PYTHON_3_9),
27
- Version.new(PYTHON_3_10),
28
- Version.new(PYTHON_3_11),
29
- Version.new(PYTHON_3_12),
30
- Version.new(PYTHON_3_13)
31
- ].freeze, T::Array[Dependabot::Version])
28
+ PRE_INSTALLED_VERSIONS_MAP = T.let(
29
+ PRE_INSTALLED_PYTHON_VERSIONS.to_h do |v|
30
+ [Dependabot::Python::Version.new(T.must(v.segments[0..1]).join(".")), v]
31
+ end,
32
+ T::Hash[Dependabot::Python::Version, Dependabot::Python::Version]
33
+ )
34
+
35
+ PRE_INSTALLED_HIGHEST_VERSION = T.let(T.must(PRE_INSTALLED_PYTHON_VERSIONS.max), Dependabot::Python::Version)
36
+
37
+ SUPPORTED_VERSIONS = T.let(
38
+ PRE_INSTALLED_PYTHON_VERSIONS.map do |v|
39
+ Dependabot::Python::Version.new(T.must(v.segments[0..1]&.join(".")))
40
+ end,
41
+ T::Array[Dependabot::Python::Version]
42
+ )
43
+
44
+ NON_SUPPORTED_HIGHEST_VERSION = "3.8"
45
+
46
+ DEPRECATED_VERSIONS = T.let([Version.new(NON_SUPPORTED_HIGHEST_VERSION)].freeze, T::Array[Dependabot::Version])
32
47
 
33
48
  sig do
34
49
  params(
35
- detected_version: String,
50
+ detected_version: T.nilable(String),
36
51
  raw_version: T.nilable(String),
37
52
  requirement: T.nilable(Requirement)
38
53
  ).void
@@ -40,7 +55,7 @@ module Dependabot
40
55
  def initialize(detected_version:, raw_version: nil, requirement: nil)
41
56
  super(
42
57
  name: LANGUAGE,
43
- detected_version: major_minor_version(detected_version),
58
+ detected_version: detected_version ? major_minor_version(detected_version) : nil,
44
59
  version: raw_version ? Version.new(raw_version) : nil,
45
60
  deprecated_versions: DEPRECATED_VERSIONS,
46
61
  supported_versions: SUPPORTED_VERSIONS,
@@ -48,25 +63,12 @@ module Dependabot
48
63
  )
49
64
  end
50
65
 
51
- sig { override.returns(T::Boolean) }
52
- def deprecated?
53
- return false unless detected_version
54
- return false if unsupported?
55
-
56
- deprecated_versions.include?(detected_version)
57
- end
58
-
59
- sig { override.returns(T::Boolean) }
60
- def unsupported?
61
- return false unless detected_version
62
-
63
- supported_versions.all? { |supported| supported > detected_version }
64
- end
65
-
66
66
  private
67
67
 
68
- sig { params(version: String).returns(Dependabot::Python::Version) }
68
+ sig { params(version: String).returns(T.nilable(Dependabot::Python::Version)) }
69
69
  def major_minor_version(version)
70
+ return nil if version.empty?
71
+
70
72
  major_minor = T.let(T.must(Version.new(version).segments[0..1]&.join(".")), String)
71
73
 
72
74
  Version.new(major_minor)
@@ -9,14 +9,6 @@ module Dependabot
9
9
  module Python
10
10
  class LanguageVersionManager
11
11
  extend T::Sig
12
- # This list must match the versions specified at the top of `python/Dockerfile`
13
- PRE_INSTALLED_PYTHON_VERSIONS = %w(
14
- 3.13.2
15
- 3.12.9
16
- 3.11.11
17
- 3.10.16
18
- 3.9.21
19
- ).freeze
20
12
 
21
13
  sig { params(python_requirement_parser: T.untyped).void }
22
14
  def initialize(python_requirement_parser:)
@@ -62,7 +54,34 @@ module Dependabot
62
54
  user_specified_python_version
63
55
  end
64
56
  else
65
- python_version_matching_imputed_requirements || PRE_INSTALLED_PYTHON_VERSIONS.first
57
+ python_version_matching_imputed_requirements || Language::PRE_INSTALLED_HIGHEST_VERSION.to_s
58
+ end
59
+ end
60
+
61
+ sig { params(requirement_string: T.nilable(String)).returns(T.nilable(String)) }
62
+ def normalize_python_exact_version(requirement_string)
63
+ return requirement_string if requirement_string.nil? || requirement_string.strip.empty?
64
+
65
+ requirement_string = requirement_string.strip
66
+
67
+ # If the requirement already has a wildcard, return nil
68
+ return nil if requirement_string == "*"
69
+
70
+ # If the requirement is not an exact version such as not X.Y.Z, =X.Y.Z, ==X.Y.Z, ===X.Y.Z
71
+ # then return the requirement as is
72
+ return requirement_string unless requirement_string.match?(/^=?={0,2}\s*\d+\.\d+(\.\d+)?(-[a-z0-9.-]+)?$/i)
73
+
74
+ parts = requirement_string.gsub(/^=+/, "").split(".")
75
+
76
+ case parts.length
77
+ when 1 # Only major version (X)
78
+ ">= #{parts[0]}.0.0 < #{parts[0].to_i + 1}.0.0" # Ensure only major version range
79
+ when 2 # Major.Minor (X.Y)
80
+ ">= #{parts[0]}.#{parts[1]}.0 < #{parts[0].to_i}.#{parts[1].to_i + 1}.0" # Ensure only minor version range
81
+ when 3 # Major.Minor.Patch (X.Y.Z)
82
+ ">= #{parts[0]}.#{parts[1]}.0 < #{parts[0].to_i}.#{parts[1].to_i + 1}.0" # Convert to >= X.Y.0
83
+ else
84
+ requirement_string
66
85
  end
67
86
  end
68
87
 
@@ -72,15 +91,22 @@ module Dependabot
72
91
 
73
92
  # If the requirement string isn't already a range (eg ">3.10"), coerce it to "major.minor.*".
74
93
  # The patch version is ignored because a non-matching patch version is unlikely to affect resolution.
75
- requirement_string = requirement_string.gsub(/\.\d+$/, ".*") if requirement_string.start_with?(/\d/)
94
+ requirement_string = requirement_string.gsub(/\.\d+$/, ".*") if /^\d/.match?(requirement_string)
95
+
96
+ requirement_string = normalize_python_exact_version(requirement_string)
97
+
98
+ if requirement_string.nil? || requirement_string.strip.empty?
99
+ return Language::PRE_INSTALLED_HIGHEST_VERSION.to_s
100
+ end
76
101
 
77
102
  # Try to match one of our pre-installed Python versions
78
103
  requirement = T.must(Python::Requirement.requirements_array(requirement_string).first)
79
- version = PRE_INSTALLED_PYTHON_VERSIONS.find { |v| requirement.satisfied_by?(Python::Version.new(v)) }
80
- return version if version
81
104
 
82
- # Otherwise we have to raise
83
- supported_versions = PRE_INSTALLED_PYTHON_VERSIONS.map { |x| x.gsub(/\.\d+$/, ".*") }.join(", ")
105
+ version = Language::PRE_INSTALLED_PYTHON_VERSIONS.find { |v| requirement.satisfied_by?(v) }
106
+ return version.to_s if version
107
+
108
+ # Otherwise we have to raise an error
109
+ supported_versions = Language::SUPPORTED_VERSIONS.map { |v| "#{v}.*" }.join(", ")
84
110
  raise ToolVersionNotSupported.new("Python", python_requirement_string, supported_versions)
85
111
  end
86
112
 
@@ -100,14 +126,13 @@ module Dependabot
100
126
 
101
127
  sig { params(requirements: T.untyped).returns(T.nilable(String)) }
102
128
  def python_version_matching(requirements)
103
- PRE_INSTALLED_PYTHON_VERSIONS.find do |version_string|
104
- version = Python::Version.new(version_string)
129
+ Language::PRE_INSTALLED_PYTHON_VERSIONS.find do |version|
105
130
  requirements.all? do |req|
106
131
  next req.any? { |r| r.satisfied_by?(version) } if req.is_a?(Array)
107
132
 
108
133
  req.satisfied_by?(version)
109
134
  end
110
- end
135
+ end.to_s
111
136
  end
112
137
  end
113
138
  end