@@ -13,6 +13,10 @@ public class PythonResolver : IPythonResolver
13
13
private readonly IPyPiClient pypiClient ;
14
14
private readonly ILogger < PythonResolver > logger ;
15
15
16
+ private readonly int maxLicenseFieldLength = 100 ;
17
+ private readonly string classifierFieldSeparator = " :: " ;
18
+ private readonly string classifierFieldLicensePrefix = "License" ;
19
+
16
20
public PythonResolver ( IPyPiClient pypiClient , ILogger < PythonResolver > logger )
17
21
{
18
22
this . pypiClient = pypiClient ;
@@ -35,7 +39,9 @@ public async Task<IList<PipGraphNode>> ResolveRootsAsync(ISingleFileComponentRec
35
39
// If we have it, we probably just want to skip at this phase as this indicates duplicates
36
40
if ( ! state . ValidVersionMap . TryGetValue ( rootPackage . Name , out _ ) )
37
41
{
38
- var result = await this . pypiClient . GetReleasesAsync ( rootPackage ) ;
42
+ var project = await this . pypiClient . GetProjectAsync ( rootPackage ) ;
43
+
44
+ var result = project . Releases ;
39
45
40
46
if ( result . Keys . Any ( ) )
41
47
{
@@ -45,7 +51,7 @@ public async Task<IList<PipGraphNode>> ResolveRootsAsync(ISingleFileComponentRec
45
51
var candidateVersion = state . ValidVersionMap [ rootPackage . Name ] . Keys . Any ( )
46
52
? state . ValidVersionMap [ rootPackage . Name ] . Keys . Last ( ) : null ;
47
53
48
- var node = new PipGraphNode ( new PipComponent ( rootPackage . Name , candidateVersion ) ) ;
54
+ var node = new PipGraphNode ( new PipComponent ( rootPackage . Name , candidateVersion , license : this . GetLicenseFromProject ( project ) , author : this . GetSupplierFromProject ( project ) ) ) ;
49
55
50
56
state . NodeReferences [ rootPackage . Name ] = node ;
51
57
@@ -103,15 +109,17 @@ private async Task<IList<PipGraphNode>> ProcessQueueAsync(ISingleFileComponentRe
103
109
else
104
110
{
105
111
// We haven't encountered this package before, so let's fetch it and find a candidate
106
- var result = await this . pypiClient . GetReleasesAsync ( dependencyNode ) ;
112
+ var project = await this . pypiClient . GetProjectAsync ( dependencyNode ) ;
113
+
114
+ var result = project . Releases ;
107
115
108
116
if ( result . Keys . Any ( ) )
109
117
{
110
118
state . ValidVersionMap [ dependencyNode . Name ] = result ;
111
119
var candidateVersion = state . ValidVersionMap [ dependencyNode . Name ] . Keys . Any ( )
112
120
? state . ValidVersionMap [ dependencyNode . Name ] . Keys . Last ( ) : null ;
113
121
114
- this . AddGraphNode ( state , state . NodeReferences [ currentNode . Name ] , dependencyNode . Name , candidateVersion ) ;
122
+ this . AddGraphNode ( state , state . NodeReferences [ currentNode . Name ] , dependencyNode . Name , candidateVersion , license : this . GetLicenseFromProject ( project ) , author : this . GetSupplierFromProject ( project ) ) ;
115
123
116
124
state . ProcessingQueue . Enqueue ( ( root , dependencyNode ) ) ;
117
125
}
@@ -155,7 +163,7 @@ private async Task<bool> InvalidateAndReprocessAsync(
155
163
156
164
var candidateVersion = state . ValidVersionMap [ pipComponent . Name ] . Keys . Any ( ) ? state . ValidVersionMap [ pipComponent . Name ] . Keys . Last ( ) : null ;
157
165
158
- node . Value = new PipComponent ( pipComponent . Name , candidateVersion ) ;
166
+ node . Value = new PipComponent ( pipComponent . Name , candidateVersion , license : pipComponent . License , author : pipComponent . Author ) ;
159
167
160
168
var dependencies = ( await this . FetchPackageDependenciesAsync ( state , newSpec ) ) . ToDictionary ( x => x . Name , x => x ) ;
161
169
@@ -201,7 +209,7 @@ private async Task<IList<PipDependencySpecification>> FetchPackageDependenciesAs
201
209
return await this . pypiClient . FetchPackageDependenciesAsync ( spec . Name , candidateVersion , packageToFetch ) ;
202
210
}
203
211
204
- private void AddGraphNode ( PythonResolverState state , PipGraphNode parent , string name , string version )
212
+ private void AddGraphNode ( PythonResolverState state , PipGraphNode parent , string name , string version , string license = null , string author = null )
205
213
{
206
214
if ( state . NodeReferences . TryGetValue ( name , out var value ) )
207
215
{
@@ -210,10 +218,57 @@ private void AddGraphNode(PythonResolverState state, PipGraphNode parent, string
210
218
}
211
219
else
212
220
{
213
- var node = new PipGraphNode ( new PipComponent ( name , version ) ) ;
221
+ var node = new PipGraphNode ( new PipComponent ( name , version , license : license , author : author ) ) ;
214
222
state . NodeReferences [ name ] = node ;
215
223
parent . Children . Add ( node ) ;
216
224
node . Parents . Add ( parent ) ;
217
225
}
218
226
}
227
+
228
+ private string GetSupplierFromProject ( PythonProject project )
229
+ {
230
+ if ( ! string . IsNullOrWhiteSpace ( project . Info ? . Maintainer ) )
231
+ {
232
+ return project . Info . Maintainer ;
233
+ }
234
+
235
+ if ( ! string . IsNullOrWhiteSpace ( project . Info ? . MaintainerEmail ) )
236
+ {
237
+ return project . Info . MaintainerEmail ;
238
+ }
239
+
240
+ if ( ! string . IsNullOrWhiteSpace ( project . Info ? . Author ) )
241
+ {
242
+ return project . Info . Author ;
243
+ }
244
+
245
+ if ( ! string . IsNullOrWhiteSpace ( project . Info ? . AuthorEmail ) )
246
+ {
247
+ return project . Info . AuthorEmail ;
248
+ }
249
+
250
+ // If none of the fields are populated, return null.
251
+ return null ;
252
+ }
253
+
254
+ private string GetLicenseFromProject ( PythonProject project )
255
+ {
256
+ // There are cases where the actual license text is found in the license field so we limit the length of this field to 100 characters.
257
+ if ( project . Info ? . License != null && project . Info ? . License . Length < this . maxLicenseFieldLength )
258
+ {
259
+ return project . Info . License ;
260
+ }
261
+
262
+ if ( project . Info ? . Classifiers != null )
263
+ {
264
+ var licenseClassifiers = project . Info . Classifiers . Where ( x => ! string . IsNullOrWhiteSpace ( x ) && x . StartsWith ( this . classifierFieldLicensePrefix ) ) ;
265
+
266
+ // Split the license classifiers by the " :: " and take the last part of the string
267
+ licenseClassifiers = licenseClassifiers . Select ( x => x . Split ( this . classifierFieldSeparator ) . Last ( ) ) . ToList ( ) ;
268
+
269
+ return string . Join ( ", " , licenseClassifiers ) ;
270
+ }
271
+
272
+ return null ;
273
+ }
219
274
}
0 commit comments