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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 3e306a8d52bca54571baf167605279980eeb625a9dcfbad1128a13de55270936
4
- data.tar.gz: 7bd9b6a85cf4e76675eaf401ed285acad0e6e518920e3cd344ae0074fa8a3a2a
3
+ metadata.gz: 26af709186ad20b222961b28c19395f53edfcab51c9fdc0e92734cd06aa2d296
4
+ data.tar.gz: c26d70a21a979a655c952c940c0f32a10f89e58cb1d1588238b64c252bf96c48
5
5
  SHA512:
6
- metadata.gz: 8a4fb9a0de4405de4e3130e78e917fc41480770e9b2053d7b368bbcfe397d202f12fde5db46edc563047322354abe87ff65a14190560f960b0c93f56e1d7a6fb
7
- data.tar.gz: 0f5f35583b805d8a5047bbea2c69db49636759c63f7afc50f0957368d99f5ae874b742ade653f6983e8c5401b34cd0d966b092c14a0ea836993a704fde5f0e63
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?(".txt", ".in") }
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
- repo_contents
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
- repo_contents(dir: relative_reqs_dir)
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
- NameNormaliser.normalise_including_extras(name, extras)
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