dependabot-python 0.299.1 → 0.301.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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: acceb8fa127937d9f47961a162388d23ff9bf1c5c0c5d0ebbbe07a65273762e7
4
- data.tar.gz: 9b8f49f1c670ea8473b506fbbfff4dfe8fb570664c2ceb07fdf9b9ff5778977a
3
+ metadata.gz: 8b08493047c96fa7ba1391e18f216a9a7456e348c0733ea17af3514d67245c04
4
+ data.tar.gz: 3a8cdb28351566647bed0354874c61dcef8e463d1cafbf1eb77cf8b5ff6585cc
5
5
  SHA512:
6
- metadata.gz: 42bc5d2b34edc189d8f0eb9c58db001695abcae2607dfb17281c512c43822c863460298a8384498746e5c0c372df850ae608437c1dc6e38a152030e36b103aea
7
- data.tar.gz: 05e37c59eeb041362d71b84e3cfe38c93a7e7ed90ef14a7bfa781dfa10795d7a778f4cf9c4a946817c3f6da5f6275505f09583eee5dceec2f0eebde3b72326a8
6
+ metadata.gz: a85078d3f128dde9d78da8b664a26be432255438e11b9fc095b48a4cb2e5a0aa8981fac701968b3b6cf87a77d160d4edb3cf69ed961afa6c24e2eb9beaf7c1c6
7
+ data.tar.gz: 041c64db083c065e6219c16b74116b00b5d443220ac6bc1c2ece40a6502af94e45b9d760fde8cb2a9a429be8c847bb7168ba388d9cccbb446fb95a6fbb7eb803
@@ -2,9 +2,9 @@ pip==24.0
2
2
  pip-tools==7.4.1
3
3
  flake8==7.1.0
4
4
  hashin==1.0.3
5
- pipenv==2024.0.2
5
+ pipenv==2024.4.1
6
6
  plette==2.1.0
7
- poetry==1.8.5
7
+ poetry==2.1.1
8
8
  # TODO: Replace 3p package `tomli` with 3.11's new stdlib `tomllib` once we drop support for Python 3.10.
9
9
  tomli==2.0.1
10
10
 
@@ -1,4 +1,4 @@
1
- # typed: true
1
+ # typed: strict
2
2
  # frozen_string_literal: true
3
3
 
4
4
  require "toml-rb"
@@ -13,12 +13,17 @@ module Dependabot
13
13
  module Python
14
14
  class FileParser
15
15
  class PythonRequirementParser
16
+ extend T::Sig
17
+
18
+ sig { returns(T::Array[Dependabot::DependencyFile]) }
16
19
  attr_reader :dependency_files
17
20
 
21
+ sig { params(dependency_files: T::Array[Dependabot::DependencyFile]).void }
18
22
  def initialize(dependency_files:)
19
23
  @dependency_files = dependency_files
20
24
  end
21
25
 
26
+ sig { returns(T::Array[String]) }
22
27
  def user_specified_requirements
23
28
  [
24
29
  pipfile_python_requirement,
@@ -32,22 +37,28 @@ module Dependabot
32
37
 
33
38
  # TODO: Add better Python version detection using dependency versions
34
39
  # (e.g., Django 2.x implies Python 3)
40
+ sig { returns(T::Array[String]) }
35
41
  def imputed_requirements
36
42
  requirement_files.flat_map do |file|
37
- file.content.lines
38
- .select { |l| l.include?(";") && l.include?("python") }
39
- .filter_map { |l| l.match(/python_version(?<req>.*?["'].*?['"])/) }
40
- .map { |re| re.named_captures.fetch("req").gsub(/['"]/, "") }
41
- .select { |r| valid_requirement?(r) }
43
+ T.must(file.content).lines
44
+ .select { |l| l.include?(";") && l.include?("python") }
45
+ .filter_map { |l| l.match(/python_version(?<req>.*?["'].*?['"])/) }
46
+ .map { |re| T.must(re.named_captures.fetch("req")).gsub(/['"]/, "") }
47
+ .select { |r| valid_requirement?(r) }
42
48
  end
43
49
  end
44
50
 
45
51
  private
46
52
 
53
+ # Parses the Pipfile content to extract the Python version requirement.
54
+ #
55
+ # @return [String, nil] the Python version requirement if specified in the Pipfile,
56
+ # or nil if the requirement is not present or does not start with a digit.
57
+ sig { returns(T.nilable(String)) }
47
58
  def pipfile_python_requirement
48
59
  return unless pipfile
49
60
 
50
- parsed_pipfile = TomlRB.parse(pipfile.content)
61
+ parsed_pipfile = TomlRB.parse(T.must(pipfile).content)
51
62
  requirement =
52
63
  parsed_pipfile.dig("requires", "python_full_version") ||
53
64
  parsed_pipfile.dig("requires", "python_version")
@@ -56,10 +67,11 @@ module Dependabot
56
67
  requirement
57
68
  end
58
69
 
70
+ sig { returns(T.nilable(String)) }
59
71
  def pyproject_python_requirement
60
72
  return unless pyproject
61
73
 
62
- pyproject_object = TomlRB.parse(pyproject.content)
74
+ pyproject_object = TomlRB.parse(T.must(pyproject).content)
63
75
 
64
76
  # Check for PEP621 requires-python
65
77
  pep621_python = pyproject_object.dig("project", "requires-python")
@@ -72,9 +84,10 @@ module Dependabot
72
84
  poetry_object&.dig("dev-dependencies", "python")
73
85
  end
74
86
 
87
+ sig { returns(T.nilable(String)) }
75
88
  def pip_compile_python_requirement
76
89
  requirement_files.each do |file|
77
- next unless pip_compile_file_matcher.lockfile_for_pip_compile_file?(file)
90
+ next unless T.must(pip_compile_file_matcher).lockfile_for_pip_compile_file?(file)
78
91
 
79
92
  marker = /^# This file is autogenerated by pip-compile with [pP]ython (?<version>\d+.\d+)$/m
80
93
  match = marker.match(file.content)
@@ -86,38 +99,41 @@ module Dependabot
86
99
  nil
87
100
  end
88
101
 
102
+ sig { returns(T.nilable(String)) }
89
103
  def python_version_file_version
90
104
  return unless python_version_file
91
105
 
92
106
  # read the content, split into lines and remove any lines with '#'
93
- content_lines = python_version_file.content.each_line.map do |line|
107
+ content_lines = T.must(T.must(python_version_file).content).each_line.map do |line|
94
108
  line.sub(/#.*$/, " ").strip
95
109
  end.reject(&:empty?)
96
110
 
97
111
  file_version = content_lines.first
98
112
  return if file_version&.empty?
99
- return unless pyenv_versions.include?("#{file_version}\n")
113
+ return unless T.must(pyenv_versions).include?("#{file_version}\n")
100
114
 
101
115
  file_version
102
116
  end
103
117
 
118
+ sig { returns(T.nilable(String)) }
104
119
  def runtime_file_python_version
105
120
  return unless runtime_file
106
121
 
107
- file_version = runtime_file.content
108
- .match(/(?<=python-).*/)&.to_s&.strip
122
+ file_version = T.must(T.must(runtime_file).content)
123
+ .match(/(?<=python-).*/)&.to_s&.strip
109
124
  return if file_version&.empty?
110
- return unless pyenv_versions.include?("#{file_version}\n")
125
+ return unless T.must(pyenv_versions).include?("#{file_version}\n")
111
126
 
112
127
  file_version
113
128
  end
114
129
 
130
+ sig { returns(T.nilable(String)) }
115
131
  def setup_file_requirement
116
132
  return unless setup_file
117
133
 
118
- req = setup_file.content
119
- .match(/python_requires\s*=\s*['"](?<req>[^'"]+)['"]/)
120
- &.named_captures&.fetch("req")&.strip
134
+ req = T.must(T.must(setup_file).content)
135
+ .match(/python_requires\s*=\s*['"](?<req>[^'"]+)['"]/)
136
+ &.named_captures&.fetch("req")&.strip
121
137
 
122
138
  requirement_class.new(req)
123
139
  req
@@ -125,22 +141,28 @@ module Dependabot
125
141
  nil
126
142
  end
127
143
 
144
+ sig { returns(T.nilable(String)) }
128
145
  def pyenv_versions
129
- @pyenv_versions ||= run_command("pyenv install --list")
146
+ @pyenv_versions = T.let(run_command("pyenv install --list"), T.nilable(String))
130
147
  end
131
148
 
149
+ sig { params(command: String, env: T::Hash[String, String]).returns(String) }
132
150
  def run_command(command, env: {})
133
151
  SharedHelpers.run_shell_command(command, env: env, stderr_to_stdout: true)
134
152
  end
135
153
 
154
+ sig { returns(T.nilable(PipCompileFileMatcher)) }
136
155
  def pip_compile_file_matcher
137
- @pip_compile_file_matcher ||= PipCompileFileMatcher.new(pip_compile_files)
156
+ @pip_compile_file_matcher = T.let(PipCompileFileMatcher.new(pip_compile_files),
157
+ T.nilable(PipCompileFileMatcher))
138
158
  end
139
159
 
160
+ sig { returns(T.class_of(Dependabot::Python::Requirement)) }
140
161
  def requirement_class
141
162
  Dependabot::Python::Requirement
142
163
  end
143
164
 
165
+ sig { params(req_string: String).returns(T::Boolean) }
144
166
  def valid_requirement?(req_string)
145
167
  requirement_class.new(req_string)
146
168
  true
@@ -148,36 +170,44 @@ module Dependabot
148
170
  false
149
171
  end
150
172
 
173
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
151
174
  def pipfile
152
175
  dependency_files.find { |f| f.name == "Pipfile" }
153
176
  end
154
177
 
178
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
155
179
  def pipfile_lock
156
180
  dependency_files.find { |f| f.name == "Pipfile.lock" }
157
181
  end
158
182
 
183
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
159
184
  def pyproject
160
185
  dependency_files.find { |f| f.name == "pyproject.toml" }
161
186
  end
162
187
 
188
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
163
189
  def setup_file
164
190
  dependency_files.find { |f| f.name == "setup.py" }
165
191
  end
166
192
 
193
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
167
194
  def python_version_file
168
195
  dependency_files.find { |f| f.name == ".python-version" }
169
196
  end
170
197
 
198
+ sig { returns(T.nilable(Dependabot::DependencyFile)) }
171
199
  def runtime_file
172
200
  dependency_files.find { |f| f.name.end_with?("runtime.txt") }
173
201
  end
174
202
 
203
+ sig { returns(T::Array[Dependabot::DependencyFile]) }
175
204
  def requirement_files
176
205
  dependency_files.select { |f| f.name.end_with?(".txt") }
177
206
  end
178
207
 
208
+ sig { returns(T::Array[DependencyFile]) }
179
209
  def pip_compile_files
180
- dependency_files.select { |f| f.name.end_with?(".in") }
210
+ @pip_compile_files ||= T.let(dependency_files.select { |f| f.name.end_with?(".in") }, T.untyped)
181
211
  end
182
212
  end
183
213
  end
@@ -489,7 +489,7 @@ module Dependabot
489
489
  @setup_cfg_file ||= T.let(get_original_file("setup.cfg"), T.nilable(Dependabot::DependencyFile))
490
490
  end
491
491
 
492
- sig { returns(T::Array[Dependabot::Python::Requirement]) }
492
+ sig { returns(T::Array[Dependabot::DependencyFile]) }
493
493
  def pip_compile_files
494
494
  @pip_compile_files ||= T.let(dependency_files.select { |f| f.name.end_with?(".in") }, T.untyped)
495
495
  end
@@ -190,7 +190,7 @@ module Dependabot
190
190
  language_version_manager.install_required_python
191
191
 
192
192
  # use system git instead of the pure Python dulwich
193
- run_poetry_command("pyenv exec poetry config experimental.system-git-client true")
193
+ run_poetry_command("pyenv exec poetry config system-git-client true")
194
194
 
195
195
  run_poetry_update_command
196
196
 
@@ -1,4 +1,4 @@
1
- # typed: true
1
+ # typed: strict
2
2
  # frozen_string_literal: true
3
3
 
4
4
  require "excon"
@@ -13,8 +13,22 @@ require "dependabot/python/name_normaliser"
13
13
  module Dependabot
14
14
  module Python
15
15
  class MetadataFinder < Dependabot::MetadataFinders::Base
16
+ extend T::Sig
16
17
  MAIN_PYPI_URL = "https://pypi.org/pypi"
17
18
 
19
+ sig do
20
+ params(
21
+ dependency: Dependabot::Dependency,
22
+ credentials: T::Array[Dependabot::Credential]
23
+ )
24
+ .void
25
+ end
26
+ def initialize(dependency:, credentials:)
27
+ super
28
+ @pypi_listing = T.let(nil, T.nilable(T::Hash[String, T.untyped]))
29
+ end
30
+
31
+ sig { returns(T.nilable(String)) }
18
32
  def homepage_url
19
33
  pypi_listing.dig("info", "home_page") ||
20
34
  pypi_listing.dig("info", "project_urls", "Homepage") ||
@@ -24,6 +38,7 @@ module Dependabot
24
38
 
25
39
  private
26
40
 
41
+ sig { override.returns(T.nilable(Dependabot::Source)) }
27
42
  def look_up_source
28
43
  potential_source_urls = [
29
44
  pypi_listing.dig("info", "project_urls", "Source"),
@@ -44,6 +59,7 @@ module Dependabot
44
59
  end
45
60
 
46
61
  # rubocop:disable Metrics/PerceivedComplexity
62
+ sig { returns(T.nilable(String)) }
47
63
  def source_from_description
48
64
  potential_source_urls = []
49
65
  desc = pypi_listing.dig("info", "description")
@@ -64,7 +80,7 @@ module Dependabot
64
80
 
65
81
  # Failing that, look for a source where the full dependency name is
66
82
  # mentioned when the link is followed
67
- @source_from_description ||=
83
+ @source_from_description ||= T.let(
68
84
  potential_source_urls.find do |url|
69
85
  full_url = Source.from_url(url)&.url
70
86
  next unless full_url
@@ -73,16 +89,19 @@ module Dependabot
73
89
  next unless response.status == 200
74
90
 
75
91
  response.body.include?(normalised_dependency_name)
76
- end
92
+ end, T.nilable(String)
93
+ )
77
94
  end
78
95
  # rubocop:enable Metrics/PerceivedComplexity
79
96
 
80
97
  # rubocop:disable Metrics/PerceivedComplexity
98
+ sig { returns(T.nilable(String)) }
81
99
  def source_from_homepage
82
- return unless homepage_body
100
+ homepage_body_local = homepage_body
101
+ return unless homepage_body_local
83
102
 
84
103
  potential_source_urls = []
85
- homepage_body.scan(Source::SOURCE_REGEX) do
104
+ homepage_body_local.scan(Source::SOURCE_REGEX) do
86
105
  potential_source_urls << Regexp.last_match.to_s
87
106
  end
88
107
 
@@ -93,7 +112,7 @@ module Dependabot
93
112
 
94
113
  return match_url if match_url
95
114
 
96
- @source_from_homepage ||=
115
+ @source_from_homepage ||= T.let(
97
116
  potential_source_urls.find do |url|
98
117
  full_url = Source.from_url(url)&.url
99
118
  next unless full_url
@@ -102,10 +121,12 @@ module Dependabot
102
121
  next unless response.status == 200
103
122
 
104
123
  response.body.include?(normalised_dependency_name)
105
- end
124
+ end, T.nilable(String)
125
+ )
106
126
  end
107
127
  # rubocop:enable Metrics/PerceivedComplexity
108
128
 
129
+ sig { returns(T.nilable(String)) }
109
130
  def homepage_body
110
131
  homepage_url = pypi_listing.dig("info", "home_page")
111
132
 
@@ -115,19 +136,21 @@ module Dependabot
115
136
  "pypi.python.org"
116
137
  ].include?(URI(homepage_url).host)
117
138
 
118
- @homepage_response ||=
139
+ @homepage_response ||= T.let(
119
140
  begin
120
141
  Dependabot::RegistryClient.get(url: homepage_url)
121
142
  rescue Excon::Error::Timeout, Excon::Error::Socket,
122
143
  Excon::Error::TooManyRedirects, ArgumentError
123
144
  nil
124
- end
145
+ end, T.nilable(Excon::Response)
146
+ )
125
147
 
126
148
  return unless @homepage_response&.status == 200
127
149
 
128
- @homepage_response.body
150
+ @homepage_response&.body
129
151
  end
130
152
 
153
+ sig { returns(T::Hash[String, T.untyped]) }
131
154
  def pypi_listing
132
155
  return @pypi_listing unless @pypi_listing.nil?
133
156
  return @pypi_listing = {} if dependency.version&.include?("+")
@@ -147,6 +170,7 @@ module Dependabot
147
170
  @pypi_listing = {} # No listing found
148
171
  end
149
172
 
173
+ sig { params(url: String).returns(Excon::Response) }
150
174
  def fetch_authed_url(url)
151
175
  if url.match(%r{(.*)://(.*?):(.*)@([^@]+)$}) &&
152
176
  Regexp.last_match&.captures&.[](1)&.include?("@")
@@ -164,6 +188,7 @@ module Dependabot
164
188
  end
165
189
  end
166
190
 
191
+ sig { returns(T::Array[String]) }
167
192
  def possible_listing_urls
168
193
  credential_urls =
169
194
  credentials
@@ -176,6 +201,7 @@ module Dependabot
176
201
  end
177
202
 
178
203
  # Strip [extras] from name (dependency_name[extra_dep,other_extra])
204
+ sig { returns(String) }
179
205
  def normalised_dependency_name
180
206
  NameNormaliser.normalise(dependency.name)
181
207
  end
@@ -268,11 +268,13 @@ module Dependabot
268
268
 
269
269
  sig do
270
270
  params(
271
- releases_json: T::Hash[String, T::Array[T::Hash[String, T.untyped]]]
271
+ releases_json: T.nilable(T::Hash[String, T::Array[T::Hash[String, T.untyped]]])
272
272
  )
273
273
  .returns(T::Array[Dependabot::Package::PackageRelease])
274
274
  end
275
275
  def format_version_releases(releases_json)
276
+ return [] unless releases_json
277
+
276
278
  releases_json.each_with_object([]) do |(version, release_data_array), versions|
277
279
  release_data = release_data_array.last
278
280
 
@@ -385,8 +387,9 @@ module Dependabot
385
387
 
386
388
  sig { params(json_url: String).returns(Excon::Response) }
387
389
  def registry_json_response_for_dependency(json_url)
390
+ url = "#{json_url.chomp('/')}/#{@dependency.name}/json"
388
391
  Dependabot::RegistryClient.get(
389
- url: "#{json_url.chomp('/')}/#{@dependency.name}/json",
392
+ url: url,
390
393
  headers: { "Accept" => APPLICATION_JSON }
391
394
  )
392
395
  end
@@ -6,7 +6,7 @@ module Dependabot
6
6
  class PipCompileFileMatcher
7
7
  extend T::Sig
8
8
 
9
- sig { params(requirements_in_files: T::Array[Dependabot::Python::Requirement]).void }
9
+ sig { params(requirements_in_files: T::Array[DependencyFile]).void }
10
10
  def initialize(requirements_in_files)
11
11
  @requirements_in_files = requirements_in_files
12
12
  end
@@ -26,7 +26,7 @@ module Dependabot
26
26
 
27
27
  private
28
28
 
29
- sig { returns(T::Array[Dependabot::Python::Requirement]) }
29
+ sig { returns(T::Array[DependencyFile]) }
30
30
  attr_reader :requirements_in_files
31
31
 
32
32
  sig { params(filename: T.any(String, Symbol)).returns(String) }
@@ -32,6 +32,11 @@ module Dependabot
32
32
  credentials: credentials
33
33
  ).fetch
34
34
  end
35
+
36
+ sig { override.returns(T::Boolean) }
37
+ def cooldown_enabled?
38
+ Dependabot::Experiments.enabled?(:enable_cooldown_for_python)
39
+ end
35
40
  end
36
41
  end
37
42
  end
@@ -11,12 +11,13 @@ module Dependabot
11
11
  class UpdateChecker
12
12
  class PipVersionResolver
13
13
  def initialize(dependency:, dependency_files:, credentials:,
14
- ignored_versions:, raise_on_ignored: false,
14
+ ignored_versions:, update_cooldown: nil, raise_on_ignored: false,
15
15
  security_advisories:)
16
16
  @dependency = dependency
17
17
  @dependency_files = dependency_files
18
18
  @credentials = credentials
19
19
  @ignored_versions = ignored_versions
20
+ @update_cooldown = update_cooldown
20
21
  @raise_on_ignored = raise_on_ignored
21
22
  @security_advisories = security_advisories
22
23
  end
@@ -50,8 +51,10 @@ module Dependabot
50
51
  credentials: credentials,
51
52
  ignored_versions: ignored_versions,
52
53
  raise_on_ignored: @raise_on_ignored,
54
+ cooldown_options: @update_cooldown,
53
55
  security_advisories: security_advisories
54
56
  )
57
+ @latest_version_finder
55
58
  end
56
59
 
57
60
  def python_requirement_parser
@@ -97,7 +97,7 @@ module Dependabot
97
97
  language_version_manager.install_required_python
98
98
 
99
99
  # use system git instead of the pure Python dulwich
100
- run_poetry_command("pyenv exec poetry config experimental.system-git-client true")
100
+ run_poetry_command("pyenv exec poetry config system-git-client true")
101
101
 
102
102
  # Shell out to Poetry, which handles everything for us.
103
103
  run_poetry_update_command
@@ -187,6 +187,7 @@ module Dependabot
187
187
  dependency_files: dependency_files,
188
188
  credentials: credentials,
189
189
  ignored_versions: ignored_versions,
190
+ update_cooldown: @update_cooldown,
190
191
  raise_on_ignored: @raise_on_ignored,
191
192
  security_advisories: security_advisories
192
193
  )
@@ -255,6 +256,7 @@ module Dependabot
255
256
  credentials: credentials,
256
257
  ignored_versions: ignored_versions,
257
258
  raise_on_ignored: @raise_on_ignored,
259
+ cooldown_options: @update_cooldown,
258
260
  security_advisories: security_advisories
259
261
  )
260
262
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: dependabot-python
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.299.1
4
+ version: 0.301.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Dependabot
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2025-02-28 00:00:00.000000000 Z
11
+ date: 2025-03-13 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: dependabot-common
@@ -16,14 +16,14 @@ dependencies:
16
16
  requirements:
17
17
  - - '='
18
18
  - !ruby/object:Gem::Version
19
- version: 0.299.1
19
+ version: 0.301.0
20
20
  type: :runtime
21
21
  prerelease: false
22
22
  version_requirements: !ruby/object:Gem::Requirement
23
23
  requirements:
24
24
  - - '='
25
25
  - !ruby/object:Gem::Version
26
- version: 0.299.1
26
+ version: 0.301.0
27
27
  - !ruby/object:Gem::Dependency
28
28
  name: debug
29
29
  requirement: !ruby/object:Gem::Requirement
@@ -291,7 +291,7 @@ licenses:
291
291
  - MIT
292
292
  metadata:
293
293
  bug_tracker_uri: https://github.com/dependabot/dependabot-core/issues
294
- changelog_uri: https://github.com/dependabot/dependabot-core/releases/tag/v0.299.1
294
+ changelog_uri: https://github.com/dependabot/dependabot-core/releases/tag/v0.301.0
295
295
  post_install_message:
296
296
  rdoc_options: []
297
297
  require_paths: