Skip to content

Commit 51367fc

Browse files
committedApr 2, 2014
requirements for directory command
1 parent f23e5c3 commit 51367fc

File tree

2 files changed

+84
-1
lines changed

2 files changed

+84
-1
lines changed
 

‎cmd/cheerio/cheerio.go

+31-1
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,29 @@
11
package main
22

33
import (
4+
"encoding/json"
45
"flag"
56
"fmt"
6-
"github.com/beyang/cheerio"
77
"log"
88
"os"
99
"strings"
1010
"sync"
11+
12+
"github.com/beyang/cheerio"
1113
)
1214

1315
const (
1416
Cmd_Repo = "repo"
1517
Cmd_Reqs = "reqs"
18+
Cmd_ReqsDir = "reqsdir"
1619
Cmd_ReqGen = "reqs-generate"
1720
Cmd_TopLevel = "toplevel"
1821
)
1922

2023
var Commands = map[string]func(args []string, flags *flag.FlagSet){
2124
Cmd_Repo: mainRepo,
2225
Cmd_Reqs: mainReqs,
26+
Cmd_ReqsDir: mainReqsDir,
2327
Cmd_ReqGen: mainReqGen,
2428
Cmd_TopLevel: mainTopLevel,
2529
}
@@ -93,6 +97,32 @@ func mainTopLevel(args []string, flags *flag.FlagSet) {
9397
}
9498
}
9599

100+
func mainReqsDir(args []string, flags *flag.FlagSet) {
101+
flags.Usage = func() {
102+
fmt.Fprintf(os.Stderr, "")
103+
flags.PrintDefaults()
104+
}
105+
flags.Parse(args[1:])
106+
if flags.NArg() < 1 {
107+
flags.Usage()
108+
os.Exit(1)
109+
}
110+
111+
dir := flags.Arg(0)
112+
reqs, err := cheerio.RequirementsForDir(dir)
113+
if err != nil {
114+
fmt.Fprintf(os.Stderr, "Error getting requirements for PyPI package directory: %s", err)
115+
os.Exit(1)
116+
}
117+
118+
// Print requirements out
119+
err = json.NewEncoder(os.Stdout).Encode(reqs)
120+
if err != nil {
121+
fmt.Fprintf(os.Stderr, "Error encoding output")
122+
os.Exit(1)
123+
}
124+
}
125+
96126
func mainReqs(args []string, flags *flag.FlagSet) {
97127
flags.Usage = func() {
98128
fmt.Fprintf(os.Stderr, "Usage: %s %s <package-name>\n", os.Args[0], args[0])

‎requirements.go

+53
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,10 @@ package cheerio
22

33
import (
44
"fmt"
5+
"io/ioutil"
56
"os"
7+
"path/filepath"
8+
"regexp"
69
"strings"
710
)
811

@@ -49,3 +52,53 @@ func ParseRequirement(reqStr string) (*Requirement, error) {
4952
Version: match[4],
5053
}, nil
5154
}
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

Comments
 (0)
Please sign in to comment.