Skip to content

Commit 0aa56cd

Browse files
authored
Merge pull request #34 from overmindtech/get-change
Implement get-change command; improve start-change/end-change with new --change and --ticket-link params
2 parents 57ea657 + 7082126 commit 0aa56cd

File tree

5 files changed

+241
-54
lines changed

5 files changed

+241
-54
lines changed

cmd/endchange.go

Lines changed: 15 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@ import (
99
"time"
1010

1111
"github.com/bufbuild/connect-go"
12-
"github.com/google/uuid"
1312
"github.com/overmindtech/ovm-cli/tracing"
1413
"github.com/overmindtech/sdp-go"
1514
log "github.com/sirupsen/logrus"
@@ -47,37 +46,38 @@ func EndChange(signals chan os.Signal, ready chan bool) int {
4746
return 1
4847
}
4948

50-
snapshotUuid, err := uuid.Parse(viper.GetString("uuid"))
51-
if err != nil {
52-
log.Errorf("invalid --uuid value '%v', error: %v", viper.GetString("uuid"), err)
53-
return 1
54-
}
55-
5649
ctx := context.Background()
5750
ctx, span := tracing.Tracer().Start(ctx, "CLI EndChange", trace.WithAttributes(
5851
attribute.String("om.config", fmt.Sprintf("%v", viper.AllSettings())),
5952
))
6053
defer span.End()
6154

62-
lf := log.Fields{
63-
"uuid": snapshotUuid.String(),
64-
}
65-
6655
ctx, err = ensureToken(ctx, signals)
6756
if err != nil {
68-
log.WithContext(ctx).WithFields(lf).WithError(err).Error("failed to authenticate")
57+
log.WithContext(ctx).WithFields(log.Fields{
58+
"url": viper.GetString("url"),
59+
}).WithError(err).Error("failed to authenticate")
6960
return 1
7061
}
7162

7263
// apply a timeout to the main body of processing
7364
ctx, cancel := context.WithTimeout(ctx, timeout)
7465
defer cancel()
7566

67+
lf := log.Fields{}
68+
changeUuid, err := getChangeUuid(ctx, sdp.ChangeStatus_CHANGE_STATUS_HAPPENING)
69+
if err != nil {
70+
log.WithError(err).WithFields(lf).Error("failed to identify change")
71+
return 1
72+
}
73+
74+
lf["uuid"] = changeUuid.String()
75+
7676
// snapClient := AuthenticatedSnapshotsClient(ctx)
7777
client := AuthenticatedChangesClient(ctx)
7878
stream, err := client.EndChange(ctx, &connect.Request[sdp.EndChangeRequest]{
7979
Msg: &sdp.EndChangeRequest{
80-
ChangeUUID: snapshotUuid[:],
80+
ChangeUUID: changeUuid[:],
8181
},
8282
})
8383
if err != nil {
@@ -100,9 +100,9 @@ func EndChange(signals chan os.Signal, ready chan bool) int {
100100
func init() {
101101
rootCmd.AddCommand(endChangeCmd)
102102

103-
endChangeCmd.PersistentFlags().String("frontend", "https://app.overmind.tech/", "The frontend base URL")
103+
withChangeUuidFlags(endChangeCmd)
104104

105-
endChangeCmd.PersistentFlags().String("uuid", "", "The UUID of the snapshot that should be displayed.")
105+
endChangeCmd.PersistentFlags().String("frontend", "https://app.overmind.tech/", "The frontend base URL")
106106

107107
endChangeCmd.PersistentFlags().String("timeout", "1m", "How long to wait for responses")
108108
}

cmd/getchange.go

Lines changed: 139 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,139 @@
1+
package cmd
2+
3+
import (
4+
"context"
5+
"encoding/json"
6+
"fmt"
7+
"os"
8+
"os/signal"
9+
"syscall"
10+
"time"
11+
12+
"github.com/bufbuild/connect-go"
13+
"github.com/google/uuid"
14+
"github.com/overmindtech/ovm-cli/tracing"
15+
"github.com/overmindtech/sdp-go"
16+
log "github.com/sirupsen/logrus"
17+
"github.com/spf13/cobra"
18+
"github.com/spf13/viper"
19+
"go.opentelemetry.io/otel/attribute"
20+
"go.opentelemetry.io/otel/trace"
21+
)
22+
23+
// getChangeCmd represents the get-change command
24+
var getChangeCmd = &cobra.Command{
25+
Use: "get-change {--uuid ID | --change https://app.overmind.tech/changes/c772d072-6b0b-4763-b7c5-ff5069beed4c}",
26+
Short: "Displays the contents of a change.",
27+
PreRun: func(cmd *cobra.Command, args []string) {
28+
// Bind these to viper
29+
err := viper.BindPFlags(cmd.Flags())
30+
if err != nil {
31+
log.WithError(err).Fatal("could not bind `get-change` flags")
32+
}
33+
},
34+
Run: func(cmd *cobra.Command, args []string) {
35+
sigs := make(chan os.Signal, 1)
36+
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
37+
38+
exitcode := GetChange(sigs, nil)
39+
tracing.ShutdownTracer()
40+
os.Exit(exitcode)
41+
},
42+
}
43+
44+
func GetChange(signals chan os.Signal, ready chan bool) int {
45+
timeout, err := time.ParseDuration(viper.GetString("timeout"))
46+
if err != nil {
47+
log.Errorf("invalid --timeout value '%v', error: %v", viper.GetString("timeout"), err)
48+
return 1
49+
}
50+
51+
ctx := context.Background()
52+
ctx, span := tracing.Tracer().Start(ctx, "CLI GetChange", trace.WithAttributes(
53+
attribute.String("om.config", fmt.Sprintf("%v", viper.AllSettings())),
54+
))
55+
defer span.End()
56+
57+
ctx, err = ensureToken(ctx, signals)
58+
if err != nil {
59+
log.WithContext(ctx).WithFields(log.Fields{
60+
"url": viper.GetString("url"),
61+
}).WithError(err).Error("failed to authenticate")
62+
return 1
63+
}
64+
65+
// apply a timeout to the main body of processing
66+
ctx, cancel := context.WithTimeout(ctx, timeout)
67+
defer cancel()
68+
69+
lf := log.Fields{}
70+
changeUuid, err := getChangeUuid(ctx, sdp.ChangeStatus(sdp.ChangeStatus_value[viper.GetString("status")]))
71+
if err != nil {
72+
log.WithError(err).WithFields(lf).Error("failed to identify change")
73+
return 1
74+
}
75+
76+
lf["uuid"] = changeUuid.String()
77+
78+
client := AuthenticatedChangesClient(ctx)
79+
response, err := client.GetChange(ctx, &connect.Request[sdp.GetChangeRequest]{
80+
Msg: &sdp.GetChangeRequest{
81+
UUID: changeUuid[:],
82+
},
83+
})
84+
if err != nil {
85+
log.WithContext(ctx).WithError(err).WithFields(log.Fields{
86+
"change-url": viper.GetString("change-url"),
87+
}).Error("failed to get change")
88+
return 1
89+
}
90+
log.WithContext(ctx).WithFields(log.Fields{
91+
"change-uuid": uuid.UUID(response.Msg.Change.Metadata.UUID),
92+
"change-created": response.Msg.Change.Metadata.CreatedAt.AsTime(),
93+
"change-name": response.Msg.Change.Properties.Title,
94+
"change-description": response.Msg.Change.Properties.Description,
95+
}).Info("found change")
96+
97+
switch viper.GetString("format") {
98+
case "json":
99+
b, _ := json.MarshalIndent(response.Msg.Change, "", " ")
100+
fmt.Println(string(b))
101+
case "markdown":
102+
changeUrl := fmt.Sprintf("%v/changes/%v", viper.GetString("frontend"), changeUuid.String())
103+
if response.Msg.Change.Metadata.NumAffectedApps != 0 || response.Msg.Change.Metadata.NumAffectedItems != 0 {
104+
// we have affected stuff
105+
fmt.Printf(`## Blast Radius &nbsp; · &nbsp; [View in Overmind](%v) <img align="center" width="16" src="" alt="chain link icon" />
106+
107+
> **Warning**
108+
> Overmind identified potentially affected apps and items as a result of this pull request.
109+
110+
<br>
111+
112+
| <img align="center" width="16" src="" alt="icon for blast radius items" /> &nbsp;Affected items |
113+
| ------------- |
114+
| [%v items](%v) |
115+
`, changeUrl, response.Msg.Change.Metadata.NumAffectedItems, changeUrl)
116+
} else {
117+
fmt.Printf(`## Blast Radius &nbsp; · &nbsp; [View in Overmind](%v) <img align="center" width="16" src="" alt="chain link icon" />
118+
119+
> **✅ Checks complete**
120+
> Overmind didn't identify any potentially affected apps and items as a result of this pull request.
121+
122+
`, changeUrl)
123+
}
124+
}
125+
126+
return 0
127+
}
128+
129+
func init() {
130+
rootCmd.AddCommand(getChangeCmd)
131+
132+
withChangeUuidFlags(getChangeCmd)
133+
getChangeCmd.PersistentFlags().String("status", "", "The expected status of the change. Use this with --ticket-link. Allowed values: CHANGE_STATUS_UNSPECIFIED, CHANGE_STATUS_DEFINING, CHANGE_STATUS_HAPPENING, CHANGE_STATUS_PROCESSING, CHANGE_STATUS_DONE")
134+
135+
getChangeCmd.PersistentFlags().String("frontend", "https://app.overmind.tech/", "The frontend base URL")
136+
getChangeCmd.PersistentFlags().String("format", "json", "How to render the change. Possible values: json, markdown")
137+
138+
getChangeCmd.PersistentFlags().String("timeout", "1m", "How long to wait for responses")
139+
}

cmd/root.go

Lines changed: 62 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import (
88
"net/http"
99
"net/url"
1010
"os"
11+
"path"
1112
"strings"
1213
"time"
1314

@@ -183,6 +184,67 @@ func ensureToken(ctx context.Context, signals chan os.Signal) (context.Context,
183184
return ctx, fmt.Errorf("no --api-key configured and target URL (%v) is insecure", parsed)
184185
}
185186

187+
// getChangeUuid returns the UUID of a change, as selected by --uuid or --change, or a state with the specified status and having --ticket-link
188+
func getChangeUuid(ctx context.Context, expectedStatus sdp.ChangeStatus) (uuid.UUID, error) {
189+
var changeUuid uuid.UUID
190+
var err error
191+
192+
if viper.GetString("uuid") != "" {
193+
changeUuid, err = uuid.Parse(viper.GetString("uuid"))
194+
if err != nil {
195+
return uuid.Nil, fmt.Errorf("invalid --uuid value '%v', error: %v", viper.GetString("uuid"), err)
196+
}
197+
}
198+
199+
if viper.GetString("change") != "" {
200+
changeUrl, err := url.ParseRequestURI(viper.GetString("change"))
201+
if err != nil {
202+
return uuid.Nil, fmt.Errorf("invalid --change value '%v', error: %v", viper.GetString("change"), err)
203+
}
204+
changeUuid, err = uuid.Parse(path.Base(changeUrl.Path))
205+
if err != nil {
206+
return uuid.Nil, fmt.Errorf("invalid --change value '%v', couldn't parse: %v", viper.GetString("change"), err)
207+
}
208+
}
209+
210+
if viper.GetString("ticket-link") != "" {
211+
client := AuthenticatedChangesClient(ctx)
212+
213+
var maybeChangeUuid *uuid.UUID
214+
changesList, err := client.ListChangesByStatus(ctx, &connect.Request[sdp.ListChangesByStatusRequest]{
215+
Msg: &sdp.ListChangesByStatusRequest{
216+
Status: expectedStatus,
217+
},
218+
})
219+
if err != nil {
220+
return uuid.Nil, errors.New("failed to searching for existing changes")
221+
}
222+
223+
for _, c := range changesList.Msg.Changes {
224+
if c.Properties.TicketLink == viper.GetString("ticket-link") {
225+
maybeChangeUuid = c.Metadata.GetUUIDParsed()
226+
if maybeChangeUuid != nil {
227+
changeUuid = *maybeChangeUuid
228+
break
229+
}
230+
}
231+
}
232+
}
233+
234+
// if changeUuid == uuid.Nil {
235+
// return uuid.Nil, errors.New("no change specified; use one of --change, --ticket-link or --uuid")
236+
// }
237+
238+
return changeUuid, nil
239+
}
240+
241+
func withChangeUuidFlags(cmd *cobra.Command) {
242+
cmd.PersistentFlags().String("change", "", "The frontend URL of the change to get")
243+
cmd.PersistentFlags().String("ticket-link", "", "Link to the ticket for this change.")
244+
cmd.PersistentFlags().String("uuid", "", "The UUID of the change that should be displayed.")
245+
cmd.MarkFlagsMutuallyExclusive("change", "ticket-link", "uuid")
246+
}
247+
186248
func init() {
187249
cobra.OnInitialize(initConfig)
188250

cmd/startchange.go

Lines changed: 15 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@ import (
99
"time"
1010

1111
"github.com/bufbuild/connect-go"
12-
"github.com/google/uuid"
1312
"github.com/overmindtech/ovm-cli/tracing"
1413
"github.com/overmindtech/sdp-go"
1514
log "github.com/sirupsen/logrus"
@@ -47,37 +46,38 @@ func StartChange(signals chan os.Signal, ready chan bool) int {
4746
return 1
4847
}
4948

50-
snapshotUuid, err := uuid.Parse(viper.GetString("uuid"))
51-
if err != nil {
52-
log.Errorf("invalid --uuid value '%v', error: %v", viper.GetString("uuid"), err)
53-
return 1
54-
}
55-
5649
ctx := context.Background()
5750
ctx, span := tracing.Tracer().Start(ctx, "CLI StartChange", trace.WithAttributes(
5851
attribute.String("om.config", fmt.Sprintf("%v", viper.AllSettings())),
5952
))
6053
defer span.End()
6154

62-
lf := log.Fields{
63-
"uuid": snapshotUuid.String(),
64-
}
65-
6655
ctx, err = ensureToken(ctx, signals)
6756
if err != nil {
68-
log.WithContext(ctx).WithFields(lf).WithError(err).Error("failed to authenticate")
57+
log.WithContext(ctx).WithFields(log.Fields{
58+
"url": viper.GetString("url"),
59+
}).WithError(err).Error("failed to authenticate")
6960
return 1
7061
}
7162

7263
// apply a timeout to the main body of processing
7364
ctx, cancel := context.WithTimeout(ctx, timeout)
7465
defer cancel()
7566

67+
lf := log.Fields{}
68+
changeUuid, err := getChangeUuid(ctx, sdp.ChangeStatus_CHANGE_STATUS_DEFINING)
69+
if err != nil {
70+
log.WithError(err).WithFields(lf).Error("failed to identify change")
71+
return 1
72+
}
73+
74+
lf["uuid"] = changeUuid.String()
75+
7676
// snapClient := AuthenticatedSnapshotsClient(ctx)
7777
client := AuthenticatedChangesClient(ctx)
7878
stream, err := client.StartChange(ctx, &connect.Request[sdp.StartChangeRequest]{
7979
Msg: &sdp.StartChangeRequest{
80-
ChangeUUID: snapshotUuid[:],
80+
ChangeUUID: changeUuid[:],
8181
},
8282
})
8383
if err != nil {
@@ -100,9 +100,9 @@ func StartChange(signals chan os.Signal, ready chan bool) int {
100100
func init() {
101101
rootCmd.AddCommand(startChangeCmd)
102102

103-
startChangeCmd.PersistentFlags().String("frontend", "https://app.overmind.tech/", "The frontend base URL")
103+
withChangeUuidFlags(startChangeCmd)
104104

105-
startChangeCmd.PersistentFlags().String("uuid", "", "The UUID of the snapshot that should be displayed.")
105+
startChangeCmd.PersistentFlags().String("frontend", "https://app.overmind.tech/", "The frontend base URL")
106106

107107
startChangeCmd.PersistentFlags().String("timeout", "1m", "How long to wait for responses")
108108
}

0 commit comments

Comments
 (0)