|
1 | 1 | import type { Packument, PackumentVersion, DependencyDepth } from '#shared/types' |
| 2 | +import { mapWithConcurrency } from '#shared/utils/async' |
2 | 3 | import { maxSatisfying } from 'semver' |
3 | 4 |
|
| 5 | +/** Concurrency limit for fetching packuments during dependency resolution */ |
| 6 | +const PACKUMENT_FETCH_CONCURRENCY = 20 |
| 7 | + |
4 | 8 | /** |
5 | 9 | * Target platform for dependency resolution. |
6 | 10 | * We resolve for linux-x64 with glibc as a representative platform. |
@@ -142,63 +146,61 @@ export async function resolveDependencyTree( |
142 | 146 | seen.add(name) |
143 | 147 | } |
144 | 148 |
|
145 | | - // Process current level in batches |
| 149 | + // Process current level with concurrency limit |
146 | 150 | const entries = [...currentLevel.entries()] |
147 | | - for (let i = 0; i < entries.length; i += 20) { |
148 | | - const batch = entries.slice(i, i + 20) |
149 | | - |
150 | | - await Promise.all( |
151 | | - batch.map(async ([name, { range, optional, path }]) => { |
152 | | - const packument = await fetchPackument(name) |
153 | | - if (!packument) return |
| 151 | + await mapWithConcurrency( |
| 152 | + entries, |
| 153 | + async ([name, { range, optional, path }]) => { |
| 154 | + const packument = await fetchPackument(name) |
| 155 | + if (!packument) return |
154 | 156 |
|
155 | | - const versions = Object.keys(packument.versions) |
156 | | - const version = resolveVersion(range, versions) |
157 | | - if (!version) return |
| 157 | + const versions = Object.keys(packument.versions) |
| 158 | + const version = resolveVersion(range, versions) |
| 159 | + if (!version) return |
158 | 160 |
|
159 | | - const versionData = packument.versions[version] |
160 | | - if (!versionData) return |
| 161 | + const versionData = packument.versions[version] |
| 162 | + if (!versionData) return |
161 | 163 |
|
162 | | - if (!matchesPlatform(versionData)) return |
| 164 | + if (!matchesPlatform(versionData)) return |
163 | 165 |
|
164 | | - const size = (versionData.dist as { unpackedSize?: number })?.unpackedSize ?? 0 |
165 | | - const key = `${name}@${version}` |
| 166 | + const size = (versionData.dist as { unpackedSize?: number })?.unpackedSize ?? 0 |
| 167 | + const key = `${name}@${version}` |
166 | 168 |
|
167 | | - // Build path for this package (path to parent + this package with version) |
168 | | - const currentPath = [...path, `${name}@${version}`] |
| 169 | + // Build path for this package (path to parent + this package with version) |
| 170 | + const currentPath = [...path, `${name}@${version}`] |
169 | 171 |
|
170 | | - if (!resolved.has(key)) { |
171 | | - const pkg: ResolvedPackage = { name, version, size, optional } |
172 | | - if (options.trackDepth) { |
173 | | - pkg.depth = level === 0 ? 'root' : level === 1 ? 'direct' : 'transitive' |
174 | | - pkg.path = currentPath |
175 | | - } |
176 | | - if (versionData.deprecated) { |
177 | | - pkg.deprecated = versionData.deprecated |
178 | | - } |
179 | | - resolved.set(key, pkg) |
| 172 | + if (!resolved.has(key)) { |
| 173 | + const pkg: ResolvedPackage = { name, version, size, optional } |
| 174 | + if (options.trackDepth) { |
| 175 | + pkg.depth = level === 0 ? 'root' : level === 1 ? 'direct' : 'transitive' |
| 176 | + pkg.path = currentPath |
180 | 177 | } |
181 | | - |
182 | | - // Collect dependencies for next level |
183 | | - if (versionData.dependencies) { |
184 | | - for (const [depName, depRange] of Object.entries(versionData.dependencies)) { |
185 | | - if (!seen.has(depName) && !nextLevel.has(depName)) { |
186 | | - nextLevel.set(depName, { range: depRange, optional: false, path: currentPath }) |
187 | | - } |
| 178 | + if (versionData.deprecated) { |
| 179 | + pkg.deprecated = versionData.deprecated |
| 180 | + } |
| 181 | + resolved.set(key, pkg) |
| 182 | + } |
| 183 | + |
| 184 | + // Collect dependencies for next level |
| 185 | + if (versionData.dependencies) { |
| 186 | + for (const [depName, depRange] of Object.entries(versionData.dependencies)) { |
| 187 | + if (!seen.has(depName) && !nextLevel.has(depName)) { |
| 188 | + nextLevel.set(depName, { range: depRange, optional: false, path: currentPath }) |
188 | 189 | } |
189 | 190 | } |
| 191 | + } |
190 | 192 |
|
191 | | - // Collect optional dependencies |
192 | | - if (versionData.optionalDependencies) { |
193 | | - for (const [depName, depRange] of Object.entries(versionData.optionalDependencies)) { |
194 | | - if (!seen.has(depName) && !nextLevel.has(depName)) { |
195 | | - nextLevel.set(depName, { range: depRange, optional: true, path: currentPath }) |
196 | | - } |
| 193 | + // Collect optional dependencies |
| 194 | + if (versionData.optionalDependencies) { |
| 195 | + for (const [depName, depRange] of Object.entries(versionData.optionalDependencies)) { |
| 196 | + if (!seen.has(depName) && !nextLevel.has(depName)) { |
| 197 | + nextLevel.set(depName, { range: depRange, optional: true, path: currentPath }) |
197 | 198 | } |
198 | 199 | } |
199 | | - }), |
200 | | - ) |
201 | | - } |
| 200 | + } |
| 201 | + }, |
| 202 | + PACKUMENT_FETCH_CONCURRENCY, |
| 203 | + ) |
202 | 204 |
|
203 | 205 | currentLevel = nextLevel |
204 | 206 | level++ |
|
0 commit comments