@@ -2,7 +2,10 @@ package cheerio
2
2
3
3
import (
4
4
"fmt"
5
+ "io/ioutil"
5
6
"os"
7
+ "path/filepath"
8
+ "regexp"
6
9
"strings"
7
10
)
8
11
@@ -49,3 +52,53 @@ func ParseRequirement(reqStr string) (*Requirement, error) {
49
52
Version : match [4 ],
50
53
}, nil
51
54
}
55
+
56
+ // Return requirements for python PyPI package in directory
57
+ func RequirementsForDir (dir string ) ([]* Requirement , error ) {
58
+ reqs := make (map [string ]* Requirement )
59
+
60
+ // If this contains a PyPI module, get requirements from PyPI graph
61
+ if pyPIName := pypiNameFromRepoDir (dir ); pyPIName != "" {
62
+ requires := DefaultPyPIGraph .Requires (pyPIName )
63
+ for _ , req := range requires {
64
+ reqs [NormalizedPkgName (req )] = & Requirement {Name : req }
65
+ }
66
+ }
67
+
68
+ // If repo contains requirements.txt, parse requirements from that (these should be more specific than those contained in a PyPIGraph, because
69
+ // they will often include version info).
70
+ reqFile := filepath .Join (dir , "requirements.txt" )
71
+ if reqFileContents , err := ioutil .ReadFile (reqFile ); err == nil {
72
+ if rawReqs , err := ParseRequirements (string (reqFileContents )); err == nil {
73
+ // Note: this currently doesn't handle pip+git-URL requirements
74
+ for _ , rawReq := range rawReqs {
75
+ reqs [NormalizedPkgName (rawReq .Name )] = rawReq
76
+ }
77
+ }
78
+ }
79
+
80
+ // if len(requirements) == 0 {
81
+ // // TODO: use depdump.py to best-effort get requirements
82
+ // }
83
+
84
+ reqList := make ([]* Requirement , 0 )
85
+ for _ , req := range reqs {
86
+ reqList = append (reqList , req )
87
+ }
88
+ return reqList , nil
89
+ }
90
+
91
+ var setupNameRegexp = regexp .MustCompile (`name\s?=\s?['"](?P<name>[A-Za-z0-9\._\-]+)['"]` )
92
+
93
+ func pypiNameFromRepoDir (dir string ) string {
94
+ setupFile := filepath .Join (dir , "setup.py" )
95
+ setupBytes , err := ioutil .ReadFile (setupFile )
96
+ if err != nil {
97
+ return ""
98
+ }
99
+ matches := setupNameRegexp .FindAllStringSubmatch (string (setupBytes ), - 1 )
100
+ if len (matches ) == 0 {
101
+ return ""
102
+ }
103
+ return matches [0 ][1 ]
104
+ }
0 commit comments