2020-03-06 04:43:28 +01:00
|
|
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"code.gitea.io/sdk/gitea"
|
|
|
|
|
|
|
|
"github.com/araddon/dateparse"
|
|
|
|
"github.com/urfave/cli/v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
// CmdTrackedTimes represents the command to operate repositories' times.
|
|
|
|
var CmdTrackedTimes = cli.Command{
|
|
|
|
Name: "times",
|
|
|
|
Aliases: []string{"time"},
|
|
|
|
Usage: "Operate on tracked times of a repository's issues & pulls",
|
|
|
|
Description: `Operate on tracked times of a repository's issues & pulls.
|
|
|
|
Depending on your permissions on the repository, only your own tracked
|
|
|
|
times might be listed.`,
|
|
|
|
ArgsUsage: "[username | #issue]",
|
|
|
|
Action: runTrackedTimes,
|
|
|
|
Subcommands: []*cli.Command{
|
|
|
|
&CmdTrackedTimesAdd,
|
|
|
|
&CmdTrackedTimesDelete,
|
|
|
|
&CmdTrackedTimesReset,
|
|
|
|
},
|
|
|
|
Flags: append([]cli.Flag{
|
|
|
|
&cli.StringFlag{
|
|
|
|
Name: "from",
|
|
|
|
Aliases: []string{"f"},
|
|
|
|
Usage: "Show only times tracked after this date",
|
|
|
|
},
|
|
|
|
&cli.StringFlag{
|
|
|
|
Name: "until",
|
|
|
|
Aliases: []string{"u"},
|
|
|
|
Usage: "Show only times tracked before this date",
|
|
|
|
},
|
|
|
|
&cli.BoolFlag{
|
|
|
|
Name: "total",
|
|
|
|
Aliases: []string{"t"},
|
|
|
|
Usage: "Print the total duration at the end",
|
|
|
|
},
|
|
|
|
}, AllDefaultFlags...),
|
|
|
|
}
|
|
|
|
|
|
|
|
func runTrackedTimes(ctx *cli.Context) error {
|
|
|
|
login, owner, repo := initCommand()
|
|
|
|
client := login.Client()
|
|
|
|
|
|
|
|
if err := client.CheckServerVersionConstraint(">= 1.11"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var times []*gitea.TrackedTime
|
|
|
|
var err error
|
|
|
|
|
|
|
|
user := ctx.Args().First()
|
|
|
|
fmt.Println(ctx.Command.ArgsUsage)
|
|
|
|
if user == "" {
|
|
|
|
// get all tracked times on the repo
|
2020-09-16 04:01:41 +02:00
|
|
|
times, _, err = client.GetRepoTrackedTimes(owner, repo)
|
2020-03-06 04:43:28 +01:00
|
|
|
} else if strings.HasPrefix(user, "#") {
|
|
|
|
// get all tracked times on the specified issue
|
2020-04-19 05:09:03 +02:00
|
|
|
issue, err := argToIndex(user)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-03-06 04:43:28 +01:00
|
|
|
}
|
2020-09-16 04:01:41 +02:00
|
|
|
times, _, err = client.ListTrackedTimes(owner, repo, issue, gitea.ListTrackedTimesOptions{})
|
2020-03-06 04:43:28 +01:00
|
|
|
} else {
|
|
|
|
// get all tracked times by the specified user
|
2020-09-16 04:01:41 +02:00
|
|
|
times, _, err = client.GetUserTrackedTimes(owner, repo, user)
|
2020-03-06 04:43:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var from, until time.Time
|
|
|
|
if ctx.String("from") != "" {
|
|
|
|
from, err = dateparse.ParseLocal(ctx.String("from"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ctx.String("until") != "" {
|
|
|
|
until, err = dateparse.ParseLocal(ctx.String("until"))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printTrackedTimes(times, outputValue, from, until, ctx.Bool("total"))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-30 08:28:58 +02:00
|
|
|
func formatDuration(seconds int64, outputType string) string {
|
|
|
|
switch outputType {
|
|
|
|
case "yaml":
|
|
|
|
case "csv":
|
|
|
|
return fmt.Sprint(seconds)
|
|
|
|
}
|
|
|
|
return time.Duration(1e9 * seconds).String()
|
|
|
|
}
|
|
|
|
|
2020-03-06 04:43:28 +01:00
|
|
|
func printTrackedTimes(times []*gitea.TrackedTime, outputType string, from, until time.Time, printTotal bool) {
|
|
|
|
var outputValues [][]string
|
|
|
|
var totalDuration int64
|
|
|
|
|
|
|
|
localLoc, err := time.LoadLocation("Local") // local timezone for time formatting
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, t := range times {
|
|
|
|
if !from.IsZero() && from.After(t.Created) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !until.IsZero() && until.Before(t.Created) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
totalDuration += t.Time
|
|
|
|
|
|
|
|
outputValues = append(
|
|
|
|
outputValues,
|
|
|
|
[]string{
|
|
|
|
t.Created.In(localLoc).Format("2006-01-02 15:04:05"),
|
|
|
|
"#" + strconv.FormatInt(t.Issue.Index, 10),
|
|
|
|
t.UserName,
|
2020-08-30 08:28:58 +02:00
|
|
|
formatDuration(t.Time, outputType),
|
2020-03-06 04:43:28 +01:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if printTotal {
|
|
|
|
outputValues = append(outputValues, []string{
|
2020-08-30 08:28:58 +02:00
|
|
|
"TOTAL", "", "", formatDuration(totalDuration, outputType),
|
2020-03-06 04:43:28 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
headers := []string{
|
|
|
|
"Created",
|
|
|
|
"Issue",
|
|
|
|
"User",
|
|
|
|
"Duration",
|
|
|
|
}
|
|
|
|
Output(outputType, headers, outputValues)
|
|
|
|
}
|
|
|
|
|
|
|
|
// CmdTrackedTimesAdd represents a sub command of times to add time to an issue
|
|
|
|
var CmdTrackedTimesAdd = cli.Command{
|
|
|
|
Name: "add",
|
|
|
|
Usage: "Track spent time on an issue",
|
|
|
|
UsageText: "tea times add <issue> <duration>",
|
|
|
|
Description: `Track spent time on an issue
|
|
|
|
Example:
|
|
|
|
tea times add 1 1h25m
|
|
|
|
`,
|
|
|
|
Action: runTrackedTimesAdd,
|
|
|
|
Flags: LoginRepoFlags,
|
|
|
|
}
|
|
|
|
|
|
|
|
func runTrackedTimesAdd(ctx *cli.Context) error {
|
|
|
|
login, owner, repo := initCommand()
|
|
|
|
|
|
|
|
if ctx.Args().Len() < 2 {
|
|
|
|
return fmt.Errorf("No issue or duration specified.\nUsage:\t%s", ctx.Command.UsageText)
|
|
|
|
}
|
|
|
|
|
2020-04-19 05:09:03 +02:00
|
|
|
issue, err := argToIndex(ctx.Args().First())
|
2020-03-06 04:43:28 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
duration, err := time.ParseDuration(strings.Join(ctx.Args().Tail(), ""))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2020-09-16 04:01:41 +02:00
|
|
|
_, _, err = login.Client().AddTime(owner, repo, issue, gitea.AddTimeOption{
|
2020-03-06 04:43:28 +01:00
|
|
|
Time: int64(duration.Seconds()),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CmdTrackedTimesDelete is a sub command of CmdTrackedTimes, and removes time from an issue
|
|
|
|
var CmdTrackedTimesDelete = cli.Command{
|
|
|
|
Name: "delete",
|
|
|
|
Aliases: []string{"rm"},
|
|
|
|
Usage: "Delete a single tracked time on an issue",
|
|
|
|
UsageText: "tea times delete <issue> <time ID>",
|
|
|
|
Action: runTrackedTimesDelete,
|
|
|
|
Flags: LoginRepoFlags,
|
|
|
|
}
|
|
|
|
|
|
|
|
func runTrackedTimesDelete(ctx *cli.Context) error {
|
|
|
|
login, owner, repo := initCommand()
|
|
|
|
client := login.Client()
|
|
|
|
|
|
|
|
if err := client.CheckServerVersionConstraint(">= 1.11"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.Args().Len() < 2 {
|
|
|
|
return fmt.Errorf("No issue or time ID specified.\nUsage:\t%s", ctx.Command.UsageText)
|
|
|
|
}
|
|
|
|
|
2020-04-19 05:09:03 +02:00
|
|
|
issue, err := argToIndex(ctx.Args().First())
|
2020-03-06 04:43:28 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
timeID, err := strconv.ParseInt(ctx.Args().Get(1), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2020-09-16 04:01:41 +02:00
|
|
|
_, err = client.DeleteTime(owner, repo, issue, timeID)
|
2020-03-06 04:43:28 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CmdTrackedTimesReset is a subcommand of CmdTrackedTimes, and
|
|
|
|
// clears all tracked times on an issue.
|
|
|
|
var CmdTrackedTimesReset = cli.Command{
|
|
|
|
Name: "reset",
|
|
|
|
Usage: "Reset tracked time on an issue",
|
|
|
|
UsageText: "tea times reset <issue>",
|
|
|
|
Action: runTrackedTimesReset,
|
|
|
|
Flags: LoginRepoFlags,
|
|
|
|
}
|
|
|
|
|
|
|
|
func runTrackedTimesReset(ctx *cli.Context) error {
|
|
|
|
login, owner, repo := initCommand()
|
|
|
|
client := login.Client()
|
|
|
|
|
|
|
|
if err := client.CheckServerVersionConstraint(">= 1.11"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.Args().Len() != 1 {
|
|
|
|
return fmt.Errorf("No issue specified.\nUsage:\t%s", ctx.Command.UsageText)
|
|
|
|
}
|
|
|
|
|
2020-04-19 05:09:03 +02:00
|
|
|
issue, err := argToIndex(ctx.Args().First())
|
2020-03-06 04:43:28 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2020-09-16 04:01:41 +02:00
|
|
|
_, err = client.ResetIssueTime(owner, repo, issue)
|
2020-03-06 04:43:28 +01:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|