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.
- checksums.yaml +4 -4
- data/lib/dependabot/python/file_fetcher.rb +1 -1
- data/lib/dependabot/python/file_updater/pip_compile_file_updater.rb +145 -69
- data/lib/dependabot/python/file_updater/pipfile_file_updater.rb +99 -20
- data/lib/dependabot/python/file_updater/pipfile_manifest_updater.rb +30 -12
- data/lib/dependabot/python/file_updater/pipfile_preparer.rb +11 -10
- data/lib/dependabot/python/file_updater/poetry_file_updater.rb +95 -24
- data/lib/dependabot/python/file_updater/pyproject_preparer.rb +21 -8
- data/lib/dependabot/python/file_updater.rb +2 -2
- data/lib/dependabot/python/language.rb +36 -34
- data/lib/dependabot/python/language_version_manager.rb +42 -17
- data/lib/dependabot/python/requirement.rb +34 -1
- metadata +7 -7
@@ -1,6 +1,7 @@
|
|
1
|
-
# typed:
|
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
|
-
|
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
|
-
|
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
|
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
|
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
|
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
|
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:
|
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
|
86
|
+
next if source_types.include?(locked_details.dig("source", "type"))
|
79
87
|
|
80
|
-
if locked_details
|
88
|
+
if locked_details.dig("source", "type") == "git"
|
81
89
|
poetry_object[key][dep_name] = {
|
82
|
-
"git" => locked_details
|
83
|
-
"rev" => locked_details
|
90
|
+
"git" => locked_details.dig("source", "url"),
|
91
|
+
"rev" => locked_details.dig("source", "reference")
|
84
92
|
}
|
85
|
-
subdirectory = locked_details
|
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
|
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
|
-
#
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
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
|
-
|
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
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
Version
|
29
|
-
|
30
|
-
|
31
|
-
|
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 ||
|
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
|
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
|
-
|
83
|
-
|
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 |
|
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
|