dependabot-uv 0.300.0 → 0.301.1
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/uv/file_fetcher.rb +51 -21
- data/lib/dependabot/uv/file_parser.rb +39 -1
- data/lib/dependabot/uv/file_updater/lock_file_updater.rb +392 -0
- data/lib/dependabot/uv/file_updater/pyproject_preparer.rb +97 -77
- data/lib/dependabot/uv/file_updater.rb +14 -1
- data/lib/dependabot/uv/package/package_details_fetcher.rb +488 -0
- data/lib/dependabot/uv/{update_checker/index_finder.rb → package/package_registry_finder.rb} +12 -6
- data/lib/dependabot/uv/update_checker/latest_version_finder.rb +15 -269
- data/lib/dependabot/uv/update_checker/lock_file_resolver.rb +48 -0
- data/lib/dependabot/uv/update_checker/pip_version_resolver.rb +7 -5
- data/lib/dependabot/uv/update_checker.rb +14 -1
- metadata +9 -6
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 26af709186ad20b222961b28c19395f53edfcab51c9fdc0e92734cd06aa2d296
|
4
|
+
data.tar.gz: c26d70a21a979a655c952c940c0f32a10f89e58cb1d1588238b64c252bf96c48
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: d0c2ac20cafd314d293a7a903947cd7fef9df95c5a85fae9f46e41d78920ae856949357cb4e6419c1804e95c19ba625c2a8fbd8bef43d9154d912c6aa31adb43
|
7
|
+
data.tar.gz: e1cb4fe17761e4f1433f9a55ae619a9271ec49359a3de0198c807f9393748633d9bfddb85002c01d782eadfc40f0a6598488fde5d9a7044e1c4e844198a26dd5
|
@@ -22,19 +22,24 @@ module Dependabot
|
|
22
22
|
CHILD_REQUIREMENT_REGEX = /^-r\s?(?<path>.*\.(?:txt|in))/
|
23
23
|
CONSTRAINT_REGEX = /^-c\s?(?<path>.*\.(?:txt|in))/
|
24
24
|
DEPENDENCY_TYPES = %w(packages dev-packages).freeze
|
25
|
+
REQUIREMENT_FILE_PATTERNS = {
|
26
|
+
extensions: [".txt", ".in"],
|
27
|
+
filenames: ["uv.lock"]
|
28
|
+
}.freeze
|
29
|
+
MAX_FILE_SIZE = 500_000
|
25
30
|
|
26
31
|
def self.required_files_in?(filenames)
|
27
|
-
return true if filenames.any? { |name| name.end_with?(
|
32
|
+
return true if filenames.any? { |name| name.end_with?(*REQUIREMENT_FILE_PATTERNS[:extensions]) }
|
28
33
|
|
29
34
|
# If there is a directory of requirements return true
|
30
35
|
return true if filenames.include?("requirements")
|
31
36
|
|
32
|
-
# If this repo is using pyproject.toml return true
|
37
|
+
# If this repo is using pyproject.toml return true (uv.lock files require a pyproject.toml)
|
33
38
|
filenames.include?("pyproject.toml")
|
34
39
|
end
|
35
40
|
|
36
41
|
def self.required_files_message
|
37
|
-
"Repo must contain a requirements.txt, requirements.in, or pyproject.toml" \
|
42
|
+
"Repo must contain a requirements.txt, uv.lock, requirements.in, or pyproject.toml" \
|
38
43
|
end
|
39
44
|
|
40
45
|
def ecosystem_versions
|
@@ -69,6 +74,7 @@ module Dependabot
|
|
69
74
|
fetched_files += requirements_in_files
|
70
75
|
fetched_files += requirement_files if requirements_txt_files.any?
|
71
76
|
|
77
|
+
fetched_files += uv_lock_files
|
72
78
|
fetched_files += project_files
|
73
79
|
fetched_files << python_version_file if python_version_file
|
74
80
|
|
@@ -125,6 +131,11 @@ module Dependabot
|
|
125
131
|
child_requirement_in_files
|
126
132
|
end
|
127
133
|
|
134
|
+
def uv_lock_files
|
135
|
+
req_txt_and_in_files.select { |f| f.name.end_with?("uv.lock") } +
|
136
|
+
child_uv_lock_files
|
137
|
+
end
|
138
|
+
|
128
139
|
def parsed_pyproject
|
129
140
|
raise "No pyproject.toml" unless pyproject
|
130
141
|
|
@@ -137,18 +148,8 @@ module Dependabot
|
|
137
148
|
return @req_txt_and_in_files if @req_txt_and_in_files
|
138
149
|
|
139
150
|
@req_txt_and_in_files = []
|
140
|
-
|
141
|
-
|
142
|
-
.select { |f| f.type == "file" }
|
143
|
-
.select { |f| f.name.end_with?(".txt", ".in") }
|
144
|
-
.reject { |f| f.size > 500_000 }
|
145
|
-
.map { |f| fetch_file_from_host(f.name) }
|
146
|
-
.select { |f| requirements_file?(f) }
|
147
|
-
.each { |f| @req_txt_and_in_files << f }
|
148
|
-
|
149
|
-
repo_contents
|
150
|
-
.select { |f| f.type == "dir" }
|
151
|
-
.each { |f| @req_txt_and_in_files += req_files_for_dir(f) }
|
151
|
+
@req_txt_and_in_files += fetch_requirement_files_from_path
|
152
|
+
@req_txt_and_in_files += fetch_requirement_files_from_dirs
|
152
153
|
|
153
154
|
@req_txt_and_in_files
|
154
155
|
end
|
@@ -158,12 +159,7 @@ module Dependabot
|
|
158
159
|
relative_reqs_dir =
|
159
160
|
requirements_dir.path.gsub(%r{^/?#{Regexp.escape(dir)}/?}, "")
|
160
161
|
|
161
|
-
|
162
|
-
.select { |f| f.type == "file" }
|
163
|
-
.select { |f| f.name.end_with?(".txt", ".in") }
|
164
|
-
.reject { |f| f.size > 500_000 }
|
165
|
-
.map { |f| fetch_file_from_host("#{relative_reqs_dir}/#{f.name}") }
|
166
|
-
.select { |f| requirements_file?(f) }
|
162
|
+
fetch_requirement_files_from_path(relative_reqs_dir)
|
167
163
|
end
|
168
164
|
|
169
165
|
def child_requirement_txt_files
|
@@ -174,6 +170,10 @@ module Dependabot
|
|
174
170
|
child_requirement_files.select { |f| f.name.end_with?(".in") }
|
175
171
|
end
|
176
172
|
|
173
|
+
def child_uv_lock_files
|
174
|
+
child_requirement_files.select { |f| f.name.end_with?("uv.lock") }
|
175
|
+
end
|
176
|
+
|
177
177
|
def child_requirement_files
|
178
178
|
@child_requirement_files ||=
|
179
179
|
begin
|
@@ -321,6 +321,36 @@ module Dependabot
|
|
321
321
|
def requirements_in_file_matcher
|
322
322
|
@requirements_in_file_matcher ||= RequiremenstFileMatcher.new(requirements_in_files)
|
323
323
|
end
|
324
|
+
|
325
|
+
def fetch_requirement_files_from_path(path = nil)
|
326
|
+
contents = path ? repo_contents(dir: path) : repo_contents
|
327
|
+
filter_requirement_files(contents, base_path: path)
|
328
|
+
end
|
329
|
+
|
330
|
+
def fetch_requirement_files_from_dirs
|
331
|
+
repo_contents
|
332
|
+
.select { |f| f.type == "dir" }
|
333
|
+
.flat_map { |dir| req_files_for_dir(dir) }
|
334
|
+
end
|
335
|
+
|
336
|
+
def filter_requirement_files(contents, base_path: nil)
|
337
|
+
contents
|
338
|
+
.select { |f| f.type == "file" }
|
339
|
+
.select { |f| file_matches_requirement_pattern?(f.name) }
|
340
|
+
.reject { |f| f.size > MAX_FILE_SIZE }
|
341
|
+
.map { |f| fetch_file_with_path(f.name, base_path) }
|
342
|
+
.select { |f| REQUIREMENT_FILE_PATTERNS[:filenames].include?(f.name) || requirements_file?(f) }
|
343
|
+
end
|
344
|
+
|
345
|
+
def file_matches_requirement_pattern?(filename)
|
346
|
+
REQUIREMENT_FILE_PATTERNS[:extensions].any? { |ext| filename.end_with?(ext) } ||
|
347
|
+
REQUIREMENT_FILE_PATTERNS[:filenames].any?(filename)
|
348
|
+
end
|
349
|
+
|
350
|
+
def fetch_file_with_path(filename, base_path)
|
351
|
+
path = base_path ? File.join(base_path, filename) : filename
|
352
|
+
fetch_file_from_host(path)
|
353
|
+
end
|
324
354
|
end
|
325
355
|
end
|
326
356
|
end
|
@@ -14,6 +14,7 @@ require "dependabot/uv/name_normaliser"
|
|
14
14
|
require "dependabot/uv/requirements_file_matcher"
|
15
15
|
require "dependabot/uv/language_version_manager"
|
16
16
|
require "dependabot/uv/package_manager"
|
17
|
+
require "toml-rb"
|
17
18
|
|
18
19
|
module Dependabot
|
19
20
|
module Uv
|
@@ -47,6 +48,7 @@ module Dependabot
|
|
47
48
|
dependency_set = DependencySet.new
|
48
49
|
|
49
50
|
dependency_set += pyproject_file_dependencies if pyproject
|
51
|
+
dependency_set += uv_lock_file_dependencies
|
50
52
|
dependency_set += requirement_dependencies if requirement_files.any?
|
51
53
|
|
52
54
|
dependency_set.dependencies
|
@@ -64,6 +66,12 @@ module Dependabot
|
|
64
66
|
)
|
65
67
|
end
|
66
68
|
|
69
|
+
# Normalize dependency names to match the PyPI index normalization
|
70
|
+
sig { params(name: String, extras: T::Array[String]).returns(String) }
|
71
|
+
def self.normalize_dependency_name(name, extras = [])
|
72
|
+
NameNormaliser.normalise_including_extras(name, extras)
|
73
|
+
end
|
74
|
+
|
67
75
|
private
|
68
76
|
|
69
77
|
sig { returns(LanguageVersionManager) }
|
@@ -164,6 +172,36 @@ module Dependabot
|
|
164
172
|
dependency_files.select { |f| f.name.end_with?(".txt", ".in") }
|
165
173
|
end
|
166
174
|
|
175
|
+
sig { returns(T::Array[DependencyFile]) }
|
176
|
+
def uv_lock_files
|
177
|
+
dependency_files.select { |f| f.name == "uv.lock" }
|
178
|
+
end
|
179
|
+
|
180
|
+
sig { returns(DependencySet) }
|
181
|
+
def uv_lock_file_dependencies
|
182
|
+
dependency_set = DependencySet.new
|
183
|
+
|
184
|
+
uv_lock_files.each do |file|
|
185
|
+
lockfile_content = TomlRB.parse(file.content)
|
186
|
+
packages = lockfile_content.fetch("package", [])
|
187
|
+
|
188
|
+
packages.each do |package_data|
|
189
|
+
next unless package_data.is_a?(Hash) && package_data["name"] && package_data["version"]
|
190
|
+
|
191
|
+
dependency_set << Dependency.new(
|
192
|
+
name: normalised_name(package_data["name"]),
|
193
|
+
version: package_data["version"],
|
194
|
+
requirements: [], # Lock files don't contain requirements
|
195
|
+
package_manager: "uv"
|
196
|
+
)
|
197
|
+
end
|
198
|
+
rescue StandardError => e
|
199
|
+
Dependabot.logger.warn("Error parsing uv.lock: #{e.message}")
|
200
|
+
end
|
201
|
+
|
202
|
+
dependency_set
|
203
|
+
end
|
204
|
+
|
167
205
|
sig { returns(DependencySet) }
|
168
206
|
def pyproject_file_dependencies
|
169
207
|
@pyproject_file_dependencies ||= T.let(PyprojectFilesParser.new(dependency_files:
|
@@ -341,7 +379,7 @@ module Dependabot
|
|
341
379
|
|
342
380
|
sig { params(name: String, extras: T::Array[String]).returns(String) }
|
343
381
|
def normalised_name(name, extras = [])
|
344
|
-
|
382
|
+
FileParser.normalize_dependency_name(name, extras)
|
345
383
|
end
|
346
384
|
|
347
385
|
sig { override.returns(T.untyped) }
|
@@ -0,0 +1,392 @@
|
|
1
|
+
# typed: true
|
2
|
+
# frozen_string_literal: true
|
3
|
+
|
4
|
+
require "toml-rb"
|
5
|
+
require "open3"
|
6
|
+
require "dependabot/dependency"
|
7
|
+
require "dependabot/shared_helpers"
|
8
|
+
require "dependabot/uv/language_version_manager"
|
9
|
+
require "dependabot/uv/version"
|
10
|
+
require "dependabot/uv/requirement"
|
11
|
+
require "dependabot/uv/file_parser/python_requirement_parser"
|
12
|
+
require "dependabot/uv/file_updater"
|
13
|
+
require "dependabot/uv/native_helpers"
|
14
|
+
require "dependabot/uv/name_normaliser"
|
15
|
+
|
16
|
+
module Dependabot
|
17
|
+
module Uv
|
18
|
+
class FileUpdater
|
19
|
+
class LockFileUpdater
|
20
|
+
require_relative "pyproject_preparer"
|
21
|
+
|
22
|
+
attr_reader :dependencies
|
23
|
+
attr_reader :dependency_files
|
24
|
+
attr_reader :credentials
|
25
|
+
attr_reader :index_urls
|
26
|
+
|
27
|
+
def initialize(dependencies:, dependency_files:, credentials:, index_urls: nil)
|
28
|
+
@dependencies = dependencies
|
29
|
+
@dependency_files = dependency_files
|
30
|
+
@credentials = credentials
|
31
|
+
@index_urls = index_urls
|
32
|
+
end
|
33
|
+
|
34
|
+
def updated_dependency_files
|
35
|
+
@updated_dependency_files ||= fetch_updated_dependency_files
|
36
|
+
end
|
37
|
+
|
38
|
+
private
|
39
|
+
|
40
|
+
def dependency
|
41
|
+
# For now, we'll only ever be updating a single dependency
|
42
|
+
dependencies.first
|
43
|
+
end
|
44
|
+
|
45
|
+
def fetch_updated_dependency_files
|
46
|
+
updated_files = []
|
47
|
+
|
48
|
+
if file_changed?(pyproject)
|
49
|
+
updated_files <<
|
50
|
+
updated_file(
|
51
|
+
file: pyproject,
|
52
|
+
content: updated_pyproject_content
|
53
|
+
)
|
54
|
+
end
|
55
|
+
|
56
|
+
if lockfile
|
57
|
+
# Use updated_lockfile_content which might raise if the lockfile doesn't change
|
58
|
+
new_content = updated_lockfile_content
|
59
|
+
raise "Expected lockfile to change!" if lockfile.content == new_content
|
60
|
+
|
61
|
+
updated_files << updated_file(file: lockfile, content: new_content)
|
62
|
+
end
|
63
|
+
|
64
|
+
updated_files
|
65
|
+
end
|
66
|
+
|
67
|
+
def updated_pyproject_content
|
68
|
+
content = pyproject.content
|
69
|
+
return content unless file_changed?(pyproject)
|
70
|
+
|
71
|
+
updated_content = content.dup
|
72
|
+
|
73
|
+
dependency.requirements.zip(dependency.previous_requirements).each do |new_r, old_r|
|
74
|
+
next unless new_r[:file] == pyproject.name && old_r[:file] == pyproject.name
|
75
|
+
|
76
|
+
updated_content = replace_dep(dependency, updated_content, new_r, old_r)
|
77
|
+
end
|
78
|
+
|
79
|
+
raise DependencyFileContentNotChanged, "Content did not change!" if content == updated_content
|
80
|
+
|
81
|
+
updated_content
|
82
|
+
end
|
83
|
+
|
84
|
+
def replace_dep(dep, content, new_r, old_r)
|
85
|
+
new_req = new_r[:requirement]
|
86
|
+
old_req = old_r[:requirement]
|
87
|
+
|
88
|
+
declaration_regex = declaration_regex(dep, old_r)
|
89
|
+
declaration_match = content.match(declaration_regex)
|
90
|
+
if declaration_match
|
91
|
+
declaration = declaration_match[:declaration]
|
92
|
+
new_declaration = declaration.sub(old_req, new_req)
|
93
|
+
content.sub(declaration, new_declaration)
|
94
|
+
else
|
95
|
+
content
|
96
|
+
end
|
97
|
+
end
|
98
|
+
|
99
|
+
def updated_lockfile_content
|
100
|
+
@updated_lockfile_content ||=
|
101
|
+
begin
|
102
|
+
original_content = lockfile.content
|
103
|
+
# Extract the original requires-python value to preserve it
|
104
|
+
original_requires_python = original_content
|
105
|
+
.match(/requires-python\s*=\s*["']([^"']+)["']/)&.captures&.first
|
106
|
+
|
107
|
+
# Use the original Python version requirement for the update if one exists
|
108
|
+
with_original_python_version(original_requires_python) do
|
109
|
+
new_lockfile = updated_lockfile_content_for(prepared_pyproject)
|
110
|
+
|
111
|
+
# Use direct string replacement to preserve the exact format
|
112
|
+
# Match the dependency section and update only the version
|
113
|
+
dependency_section_pattern = /
|
114
|
+
(\[\[package\]\]\s*\n
|
115
|
+
.*?name\s*=\s*["']#{Regexp.escape(dependency.name)}["']\s*\n
|
116
|
+
.*?)
|
117
|
+
(version\s*=\s*["'][^"']+["'])
|
118
|
+
(.*?)
|
119
|
+
(\[\[package\]\]|\z)
|
120
|
+
/xm
|
121
|
+
|
122
|
+
result = original_content.sub(dependency_section_pattern) do
|
123
|
+
section_start = Regexp.last_match(1)
|
124
|
+
version_line = "version = \"#{dependency.version}\""
|
125
|
+
section_end = Regexp.last_match(3)
|
126
|
+
next_section_or_end = Regexp.last_match(4)
|
127
|
+
|
128
|
+
"#{section_start}#{version_line}#{section_end}#{next_section_or_end}"
|
129
|
+
end
|
130
|
+
|
131
|
+
# If the content didn't change and we expect it to, something went wrong
|
132
|
+
if result == original_content
|
133
|
+
Dependabot.logger.warn("Package section not found for #{dependency.name}, falling back to raw update")
|
134
|
+
result = new_lockfile
|
135
|
+
end
|
136
|
+
|
137
|
+
# Restore the original requires-python if it exists
|
138
|
+
if original_requires_python
|
139
|
+
result = result.gsub(/requires-python\s*=\s*["'][^"']+["']/,
|
140
|
+
"requires-python = \"#{original_requires_python}\"")
|
141
|
+
end
|
142
|
+
|
143
|
+
result
|
144
|
+
end
|
145
|
+
end
|
146
|
+
end
|
147
|
+
|
148
|
+
# Helper method to temporarily override Python version during operations
|
149
|
+
def with_original_python_version(original_requires_python)
|
150
|
+
if original_requires_python
|
151
|
+
original_python_version = @original_python_version
|
152
|
+
@original_python_version = original_requires_python
|
153
|
+
result = yield
|
154
|
+
@original_python_version = original_python_version
|
155
|
+
result
|
156
|
+
else
|
157
|
+
yield
|
158
|
+
end
|
159
|
+
end
|
160
|
+
|
161
|
+
def prepared_pyproject
|
162
|
+
@prepared_pyproject ||=
|
163
|
+
begin
|
164
|
+
content = updated_pyproject_content
|
165
|
+
content = sanitize(content)
|
166
|
+
content = freeze_other_dependencies(content)
|
167
|
+
content = update_python_requirement(content)
|
168
|
+
content
|
169
|
+
end
|
170
|
+
end
|
171
|
+
|
172
|
+
def freeze_other_dependencies(pyproject_content)
|
173
|
+
PyprojectPreparer
|
174
|
+
.new(pyproject_content: pyproject_content, lockfile: lockfile)
|
175
|
+
.freeze_top_level_dependencies_except(dependencies)
|
176
|
+
end
|
177
|
+
|
178
|
+
def update_python_requirement(pyproject_content)
|
179
|
+
PyprojectPreparer
|
180
|
+
.new(pyproject_content: pyproject_content)
|
181
|
+
.update_python_requirement(language_version_manager.python_version)
|
182
|
+
end
|
183
|
+
|
184
|
+
def sanitize(pyproject_content)
|
185
|
+
PyprojectPreparer
|
186
|
+
.new(pyproject_content: pyproject_content)
|
187
|
+
.sanitize
|
188
|
+
end
|
189
|
+
|
190
|
+
def updated_lockfile_content_for(pyproject_content)
|
191
|
+
SharedHelpers.in_a_temporary_directory do
|
192
|
+
SharedHelpers.with_git_configured(credentials: credentials) do
|
193
|
+
write_temporary_dependency_files(pyproject_content)
|
194
|
+
|
195
|
+
# Install Python before writing .python-version to make sure we use a version that's available
|
196
|
+
language_version_manager.install_required_python
|
197
|
+
|
198
|
+
# Determine the Python version to use after installation
|
199
|
+
python_version = determine_python_version
|
200
|
+
|
201
|
+
# Now write the .python-version file with a version we know is installed
|
202
|
+
File.write(".python-version", python_version)
|
203
|
+
|
204
|
+
run_update_command
|
205
|
+
|
206
|
+
File.read("uv.lock")
|
207
|
+
end
|
208
|
+
end
|
209
|
+
end
|
210
|
+
|
211
|
+
def run_update_command
|
212
|
+
command = "pyenv exec uv lock --upgrade-package #{dependency.name}"
|
213
|
+
fingerprint = "pyenv exec uv lock --upgrade-package <dependency_name>"
|
214
|
+
|
215
|
+
run_command(command, fingerprint:)
|
216
|
+
end
|
217
|
+
|
218
|
+
def run_command(command, fingerprint: nil)
|
219
|
+
SharedHelpers.run_shell_command(command, fingerprint: fingerprint)
|
220
|
+
end
|
221
|
+
|
222
|
+
def write_temporary_dependency_files(pyproject_content)
|
223
|
+
dependency_files.each do |file|
|
224
|
+
path = file.name
|
225
|
+
FileUtils.mkdir_p(Pathname.new(path).dirname)
|
226
|
+
File.write(path, file.content)
|
227
|
+
end
|
228
|
+
|
229
|
+
# Only write the .python-version file after the language version manager has
|
230
|
+
# installed the required Python version to ensure it's available
|
231
|
+
# Overwrite the pyproject with updated content
|
232
|
+
File.write("pyproject.toml", pyproject_content)
|
233
|
+
end
|
234
|
+
|
235
|
+
def determine_python_version
|
236
|
+
# Check available Python versions through pyenv
|
237
|
+
available_versions = nil
|
238
|
+
begin
|
239
|
+
available_versions = SharedHelpers.run_shell_command("pyenv versions --bare")
|
240
|
+
.split("\n")
|
241
|
+
.map(&:strip)
|
242
|
+
.reject(&:empty?)
|
243
|
+
rescue StandardError => e
|
244
|
+
Dependabot.logger.warn("Error checking available Python versions: #{e}")
|
245
|
+
end
|
246
|
+
|
247
|
+
# Try to find the closest match for our priority order
|
248
|
+
preferred_version = find_preferred_version(available_versions)
|
249
|
+
|
250
|
+
if preferred_version
|
251
|
+
# Just return the major.minor version string
|
252
|
+
preferred_version.match(/^(\d+\.\d+)/)[1]
|
253
|
+
else
|
254
|
+
# If all else fails, use "system" which should work with whatever Python is available
|
255
|
+
"system"
|
256
|
+
end
|
257
|
+
end
|
258
|
+
|
259
|
+
def find_preferred_version(available_versions)
|
260
|
+
return nil unless available_versions&.any?
|
261
|
+
|
262
|
+
# Try each strategy in order of preference
|
263
|
+
try_version_from_file(available_versions) ||
|
264
|
+
try_version_from_requires_python(available_versions) ||
|
265
|
+
try_highest_python3_version(available_versions)
|
266
|
+
end
|
267
|
+
|
268
|
+
def try_version_from_file(available_versions)
|
269
|
+
python_version_file = dependency_files.find { |f| f.name == ".python-version" }
|
270
|
+
return nil unless python_version_file && !python_version_file.content.strip.empty?
|
271
|
+
|
272
|
+
requested_version = python_version_file.content.strip
|
273
|
+
return requested_version if version_available?(available_versions, requested_version)
|
274
|
+
|
275
|
+
Dependabot.logger.info("Python version #{requested_version} from .python-version not available")
|
276
|
+
nil
|
277
|
+
end
|
278
|
+
|
279
|
+
def try_version_from_requires_python(available_versions)
|
280
|
+
return nil unless @original_python_version
|
281
|
+
|
282
|
+
version_match = @original_python_version.match(/(\d+\.\d+)/)
|
283
|
+
return nil unless version_match
|
284
|
+
|
285
|
+
requested_version = version_match[1]
|
286
|
+
return requested_version if version_available?(available_versions, requested_version)
|
287
|
+
|
288
|
+
Dependabot.logger.info("Python version #{requested_version} from requires-python not available")
|
289
|
+
nil
|
290
|
+
end
|
291
|
+
|
292
|
+
def try_highest_python3_version(available_versions)
|
293
|
+
python3_versions = available_versions
|
294
|
+
.select { |v| v.match(/^3\.\d+/) }
|
295
|
+
.sort_by { |v| Gem::Version.new(v.match(/^(\d+\.\d+)/)[1]) }
|
296
|
+
.reverse
|
297
|
+
|
298
|
+
python3_versions.first # returns nil if array is empty
|
299
|
+
end
|
300
|
+
|
301
|
+
def version_available?(available_versions, requested_version)
|
302
|
+
# Check if the exact version or a version with the same major.minor is available
|
303
|
+
available_versions.any? do |v|
|
304
|
+
v == requested_version || v.start_with?("#{requested_version}.")
|
305
|
+
end
|
306
|
+
end
|
307
|
+
|
308
|
+
def sanitize_env_name(url)
|
309
|
+
url.gsub(%r{^https?://}, "").gsub(/[^a-zA-Z0-9]/, "_").upcase
|
310
|
+
end
|
311
|
+
|
312
|
+
def declaration_regex(dep, old_req)
|
313
|
+
escaped_name = Regexp.escape(dep.name)
|
314
|
+
# Extract the requirement operator and version
|
315
|
+
operator = old_req.fetch(:requirement).match(/^(.+?)[0-9]/)&.captures&.first
|
316
|
+
# Escape special regex characters in the operator
|
317
|
+
escaped_operator = Regexp.escape(operator) if operator
|
318
|
+
|
319
|
+
# Match various formats of dependency declarations:
|
320
|
+
# 1. "dependency==1.0.0" (with quotes around the entire string)
|
321
|
+
# 2. dependency==1.0.0 (without quotes)
|
322
|
+
# The declaration should only include the package name, operator, and version
|
323
|
+
# without the enclosing quotes
|
324
|
+
/
|
325
|
+
["']?(?<declaration>#{escaped_name}\s*#{escaped_operator}[\d\.\*]+)["']?
|
326
|
+
/x
|
327
|
+
end
|
328
|
+
|
329
|
+
def escape(name)
|
330
|
+
Regexp.escape(name).gsub("\\-", "[-_.]")
|
331
|
+
end
|
332
|
+
|
333
|
+
def file_changed?(file)
|
334
|
+
return false unless file
|
335
|
+
|
336
|
+
dependencies.any? do |dep|
|
337
|
+
dep.requirements.any? { |r| r[:file] == file.name } &&
|
338
|
+
requirement_changed?(file, dep)
|
339
|
+
end
|
340
|
+
end
|
341
|
+
|
342
|
+
def requirement_changed?(file, dependency)
|
343
|
+
changed_requirements =
|
344
|
+
dependency.requirements - dependency.previous_requirements
|
345
|
+
|
346
|
+
changed_requirements.any? { |f| f[:file] == file.name }
|
347
|
+
end
|
348
|
+
|
349
|
+
def updated_file(file:, content:)
|
350
|
+
updated_file = file.dup
|
351
|
+
updated_file.content = content
|
352
|
+
updated_file
|
353
|
+
end
|
354
|
+
|
355
|
+
def normalise(name)
|
356
|
+
NameNormaliser.normalise(name)
|
357
|
+
end
|
358
|
+
|
359
|
+
def python_requirement_parser
|
360
|
+
@python_requirement_parser ||=
|
361
|
+
FileParser::PythonRequirementParser.new(
|
362
|
+
dependency_files: dependency_files
|
363
|
+
)
|
364
|
+
end
|
365
|
+
|
366
|
+
def language_version_manager
|
367
|
+
@language_version_manager ||=
|
368
|
+
LanguageVersionManager.new(
|
369
|
+
python_requirement_parser: python_requirement_parser
|
370
|
+
)
|
371
|
+
end
|
372
|
+
|
373
|
+
def pyproject
|
374
|
+
@pyproject ||=
|
375
|
+
dependency_files.find { |f| f.name == "pyproject.toml" }
|
376
|
+
end
|
377
|
+
|
378
|
+
def lockfile
|
379
|
+
@lockfile ||= uv_lock
|
380
|
+
end
|
381
|
+
|
382
|
+
def python_helper_path
|
383
|
+
NativeHelpers.python_helper_path
|
384
|
+
end
|
385
|
+
|
386
|
+
def uv_lock
|
387
|
+
dependency_files.find { |f| f.name == "uv.lock" }
|
388
|
+
end
|
389
|
+
end
|
390
|
+
end
|
391
|
+
end
|
392
|
+
end
|