func (a BugApplication) List(args []string) { issues, _ := ioutil.ReadDir(string(bugs.GetRootDir()) + "/issues") // No parameters, print a list of all bugs if len(args) == 0 { for idx, issue := range issues { var dir bugs.Directory = bugs.Directory(issue.Name()) fmt.Printf("Issue %d: %s\n", idx+1, dir.ToTitle()) } return } // There were parameters, so show the full description of each // of those issues b := bugs.Bug{} for i, length := 0, len(args); i < length; i += 1 { idx, err := strconv.Atoi(args[i]) if err != nil { listTags(issues, args) return } if idx > len(issues) || idx < 1 { fmt.Printf("Invalid issue number %d\n", idx) continue } if err == nil { b.LoadBug(bugs.Directory(bugs.GetRootDir() + "/issues/" + bugs.Directory(issues[idx-1].Name()))) b.ViewBug() if i < length-1 { fmt.Printf("\n--\n\n") } } } fmt.Printf("\n") }
func (m BugPageHandler) Get(r *http.Request, extras map[string]interface{}) (string, error) { if r.URL.Path == "/issues" || r.URL.Path == "/issues/" { return getBugList() } bugDir := string(bugs.GetRootDir()) + r.URL.Path b := bugs.Bug{} b.LoadBug(bugs.Directory(bugDir)) switch r.URL.Query().Get("format") { case "json": bJSON, _ := json.Marshal(b) return string(bJSON), nil default: page := BugRenderer{Bug: b} page.RootElement = "RBugPage" page.Title = b.Title("") page.JSFiles = []string{ // Bootstrap JS //"https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/js/bootstrap.min.js", // React JS "https://cdnjs.cloudflare.com/ajax/libs/react/0.14.3/react.js", "https://cdnjs.cloudflare.com/ajax/libs/react/0.14.3/react-dom.js", "/js/BugPage.js", } page.CSSFiles = []string{ "https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css", "https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap-theme.min.css"} return HTMLPageRenderer.Render(page), nil } }
func getBugName(b bugs.Bug, idx int) string { if id := b.Identifier(); id != "" { return fmt.Sprintf("Issue %s", id) } else { return fmt.Sprintf("Issue %d", idx+1) } }
func (a BugApplication) Roadmap() { issues, _ := ioutil.ReadDir(string(bugs.GetRootDir()) + "/issues") var bgs [](bugs.Bug) for idx, _ := range issues { b := bugs.Bug{} b.LoadBug(bugs.Directory(bugs.GetRootDir() + "/issues/" + bugs.Directory(issues[idx].Name()))) bgs = append(bgs, b) } sort.Sort(BugListByMilestone(bgs)) fmt.Printf("# Roadmap for %s\n", bugs.GetRootDir().GetShortName().ToTitle()) milestone := "" for i := len(bgs) - 1; i >= 0; i -= 1 { b := bgs[i] newMilestone := b.Milestone() if milestone != newMilestone { if newMilestone == "" { fmt.Printf("\n## No milestone set:\n") } else { fmt.Printf("\n## %s:\n", newMilestone) } } fmt.Printf("- %s\n", b.Title) milestone = newMilestone } }
func List(args ArgumentList, stdout *os.File) { issues, _ := ioutil.ReadDir(string(bugs.GetIssuesDir())) var wantTags bool = false if args.HasArgument("--tags") { wantTags = true } // No parameters, print a list of all bugs if len(args) == 0 || (wantTags && len(args) == 1) { //os.Stdout = stdout for idx, issue := range issues { if issue.IsDir() != true { continue } var dir bugs.Directory = bugs.GetIssuesDir() + bugs.Directory(issue.Name()) b := bugs.Bug{dir} name := getBugName(b, idx) if wantTags == false { fmt.Printf("%s: %s\n", name, b.Title("")) } else { fmt.Printf("%s: %s\n", name, b.Title("tags")) } } return } // getAllTags() is defined in Tag.go // Get a list of tags, so that when we encounter // an error we can check if it's because the user // provided a tagname instead of a BugID. If they // did, then list bugs matching that tag instead // of full descriptions tags := getAllTags() // There were parameters, so show the full description of each // of those issues for i, length := 0, len(args); i < length; i += 1 { b, err := bugs.LoadBugByHeuristic(args[i]) if err != nil { for _, tagname := range tags { if tagname == args[i] { listTags(issues, args) return } } fmt.Printf("%s\n", err.Error()) continue } b.ViewBug() if i < length-1 { fmt.Printf("\n--\n\n") } } fmt.Printf("\n") }
func (a BugApplication) List(args ArgumentList) { issues, _ := ioutil.ReadDir(string(bugs.GetIssuesDir())) var wantTags bool = false if args.HasArgument("--tags") { wantTags = true } // No parameters, print a list of all bugs if len(args) == 0 || (wantTags && len(args) == 1) { for idx, issue := range issues { if issue.IsDir() != true { continue } var dir bugs.Directory = bugs.GetIssuesDir() + bugs.Directory(issue.Name()) b := bugs.Bug{dir} if wantTags == false { fmt.Printf("Issue %d: %s\n", idx+1, b.Title("")) } else { fmt.Printf("Issue %d: %s\n", idx+1, b.Title("tags")) } } return } // There were parameters, so show the full description of each // of those issues b := bugs.Bug{} for i, length := 0, len(args); i < length; i += 1 { idx, err := strconv.Atoi(args[i]) if err != nil { listTags(issues, args) return } if idx > len(issues) || idx < 1 { fmt.Printf("Invalid issue number %d\n", idx) continue } if err == nil { b.LoadBug(bugs.Directory(bugs.GetIssuesDir() + bugs.Directory(issues[idx-1].Name()))) b.ViewBug() if i < length-1 { fmt.Printf("\n--\n\n") } } } fmt.Printf("\n") }
func (a BugApplication) Create(Args []string) { if len(Args) < 1 || (len(Args) < 2 && Args[0] == "-n") { fmt.Printf("Usage: %s create [-n] Bug Description\n", os.Args[0]) fmt.Printf("\nNo Bug Description provided.\n") return } var noDesc bool = false if Args != nil && Args[0] == "-n" { noDesc = true Args = Args[1:] } var bug bugs.Bug bug = bugs.Bug{ Title: strings.Join(Args, " "), } dir, _ := bug.GetDirectory() fmt.Printf("Created issue: %s\n", bug.Title) var mode os.FileMode mode = 0775 os.Mkdir(string(dir), mode) if noDesc { txt := []byte("") ioutil.WriteFile(string(dir)+"/Description", txt, 0644) } else { cmd := exec.Command(getEditor(), string(dir)+"/Description") cmd.Stdin = os.Stdin cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr err := cmd.Run() if err != nil { log.Fatal(err) } } }
func listTags(files []os.FileInfo, args []string) { hasTag := func(tags []string, tag string) bool { for _, candidate := range tags { if candidate == tag { return true } } return false } b := bugs.Bug{} for idx, _ := range files { b.LoadBug(bugs.Directory(bugs.GetRootDir() + "/issues/" + bugs.Directory(files[idx].Name()))) tags := b.StringTags() for _, tag := range args { if hasTag(tags, tag) { fmt.Printf("Issue %d: %s (%s)\n", idx+1, b.Title, strings.Join(tags, ", ")) } } } }
func listTags(files []os.FileInfo, args ArgumentList) { b := bugs.Bug{} for idx, _ := range files { b.LoadBug(bugs.Directory(bugs.GetIssuesDir() + bugs.Directory(files[idx].Name()))) for _, tag := range args { if b.HasTag(bugs.Tag(tag)) { fmt.Printf("%s: %s\n", getBugName(b, idx), b.Title("tags")) } } } }
func Create(Args ArgumentList) { if len(Args) < 1 || (len(Args) < 2 && Args[0] == "-n") { fmt.Fprintf(os.Stderr, "Usage: %s create [-n] Bug Description\n", os.Args[0]) fmt.Fprintf(os.Stderr, "\nNo Bug Description provided.\n") return } var noDesc bool = false if Args.HasArgument("-n") { noDesc = true Args = Args[1:] } Args, argVals := Args.GetAndRemoveArguments([]string{"--tag", "--status", "--priority", "--milestone", "--identifier"}) tag := argVals[0] status := argVals[1] priority := argVals[2] milestone := argVals[3] identifier := argVals[4] if Args.HasArgument("--generate-id") { for i, token := range Args { if token == "--generate-id" { if i+1 < len(Args) { Args = append(Args[:i], Args[i+1:]...) break } else { Args = Args[:i] break } } } identifier = generateID(strings.Join(Args, " ")) } var bug bugs.Bug bug = bugs.Bug{ Dir: bugs.GetIssuesDir() + bugs.TitleToDir(strings.Join(Args, " ")), } dir := bug.GetDirectory() var mode os.FileMode mode = 0775 os.Mkdir(string(dir), mode) if noDesc { txt := []byte("") ioutil.WriteFile(string(dir)+"/Description", txt, 0644) } else { cmd := exec.Command(getEditor(), string(dir)+"/Description") cmd.Stdin = os.Stdin cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr err := cmd.Run() if err != nil { log.Fatal(err) } } if tag != "" { bug.TagBug(bugs.Tag(tag)) } if status != "" { bug.SetStatus(status) } if priority != "" { bug.SetPriority(priority) } if milestone != "" { bug.SetMilestone(milestone) } if identifier != "" { bug.SetIdentifier(identifier) } fmt.Printf("Created issue: %s\n", bug.Title("")) }
func beImportBug(identifier, issuesDir, fullbepath string) { /* BE appears to store the top level data of a bug ins a json file named values with the format: { "creator": "Dave MacFarlane <*****@*****.**>", "reporter": "Dave MacFarlane <*****@*****.**>", "severity": "minor", "status": "open", "summary": "abc", "time": "Tue, 12 Jan 2016 00:05:28 +0000" } and the description of bugs entirely in comments. All we really care about is the summary so that we can get the directory name for the issues/ directory, but the severity+status can also be used as a status to ensure that we have at least 1 file to be tracked by git. */ type BeValues struct { Creator string `json:creator` Reporter string `json:reporter` Severity string `json:severity` Status string `json:status` Summary string `json:summary` Time string `json:time` } file := fullbepath + "/values" fmt.Printf("File: %s\n", file) data, _ := ioutil.ReadFile(file) var beBug BeValues err := json.Unmarshal([]byte(data), &beBug) if err != nil { fmt.Printf("Error unmarshalling data: %s\n", err.Error()) } fmt.Printf("%s\n", beBug) bugdir := bugs.TitleToDir(beBug.Summary) b := bugs.Bug{bugs.Directory(issuesDir) + bugdir} if dir := b.GetDirectory(); dir != "" { os.Mkdir(string(dir), 0755) } if beBug.Status != "" && beBug.Severity != "" { b.SetStatus(beBug.Status + ":" + beBug.Severity) } comments := fullbepath + "/comments/" dir, err := os.Open(comments) files, err := dir.Readdir(-1) var Description string if len(files) > 0 && err == nil { for _, file := range files { if file.IsDir() { Description = Description + "\n" + beImportComments(b, comments+file.Name(), len(files) > 1) } } } b.SetDescription(Description) b.SetIdentifier(identifier) }