Skip to content

Commit 3b5cdae

Browse files
authored
CLOUDP-197021: Atlas deployments logs (#2350)
1 parent 4c06b4b commit 3b5cdae

File tree

5 files changed

+218
-6
lines changed

5 files changed

+218
-6
lines changed

docs/atlascli/command/atlas-deployments-logs.txt

+28
Original file line numberDiff line numberDiff line change
@@ -39,14 +39,42 @@ Options
3939
- string
4040
- false
4141
- Name of the deployment.
42+
* - --end
43+
- int
44+
- false
45+
- UNIX Epoch-formatted ending date and time for the range of log messages to retrieve. This value defaults to the current timestamp.
46+
* - --force
47+
-
48+
- false
49+
- Overwrites the destination file.
4250
* - -h, --help
4351
-
4452
- false
4553
- help for logs
54+
* - --hostname
55+
- string
56+
- false
57+
- Name of the host that stores the log files that you want to download.
58+
* - --name
59+
- string
60+
- false
61+
- Name of the log file (e.g. mongodb.gz|mongos.gz|mongosqld.gz|mongodb-audit-log.gz|mongos-audit-log.gz).
4662
* - -o, --output
4763
- string
4864
- false
4965
- Output format. Valid values are json, json-path, go-template, or go-template-file. To see full output, use the -o json option.
66+
* - --projectId
67+
- string
68+
- false
69+
- Hexadecimal string that identifies the project to use. This option overrides the settings in the configuration file or environment variable.
70+
* - --start
71+
- int
72+
- false
73+
- UNIX Epoch-formatted starting date and time for the range of log messages to retrieve. This value defaults to 24 hours prior to the current timestamp.
74+
* - --type
75+
- string
76+
- false
77+
- Type of deployment that you want to create. Valid values are ATLAS or LOCAL.
5078

5179
Inherited Options
5280
-----------------

internal/cli/atlas/deployments/logs.go

+130-4
Original file line numberDiff line numberDiff line change
@@ -15,48 +15,163 @@
1515
package deployments
1616

1717
import (
18+
"compress/gzip"
1819
"context"
1920
"errors"
21+
"fmt"
22+
"io"
23+
"os"
24+
"path/filepath"
2025
"strings"
2126

2227
"github.com/mongodb/mongodb-atlas-cli/internal/cli"
2328
"github.com/mongodb/mongodb-atlas-cli/internal/cli/atlas/deployments/options"
2429
"github.com/mongodb/mongodb-atlas-cli/internal/cli/require"
30+
"github.com/mongodb/mongodb-atlas-cli/internal/config"
2531
"github.com/mongodb/mongodb-atlas-cli/internal/flag"
26-
"github.com/mongodb/mongodb-atlas-cli/internal/log"
32+
"github.com/mongodb/mongodb-atlas-cli/internal/search"
33+
"github.com/mongodb/mongodb-atlas-cli/internal/store"
2734
"github.com/mongodb/mongodb-atlas-cli/internal/usage"
35+
"github.com/spf13/afero"
2836
"github.com/spf13/cobra"
37+
"go.mongodb.org/atlas-sdk/v20230201008/admin"
2938
)
3039

3140
type DownloadOpts struct {
3241
cli.OutputOpts
3342
cli.GlobalOpts
43+
cli.DownloaderOpts
3444
options.DeploymentOpts
45+
downloadStore store.LogsDownloader
46+
host string
47+
name string
48+
start int64
49+
end int64
3550
}
3651

3752
var ErrAtlasNotSupported = errors.New("atlas deployments are not supported")
3853

54+
const filePermission = 0644
55+
56+
func (opts *DownloadOpts) initStore(ctx context.Context) func() error {
57+
return func() error {
58+
var err error
59+
opts.downloadStore, err = store.New(store.AuthenticatedPreset(config.Default()), store.WithContext(ctx))
60+
return err
61+
}
62+
}
63+
3964
func (opts *DownloadOpts) Run(ctx context.Context) error {
4065
if _, err := opts.SelectDeployments(ctx, opts.ConfigProjectID()); err != nil {
4166
return err
4267
}
4368

69+
if opts.IsLocalDeploymentType() {
70+
return opts.RunLocal(ctx)
71+
}
72+
4473
if opts.IsAtlasDeploymentType() {
45-
return ErrAtlasNotSupported
74+
if err := opts.validateAtlasFlags(); err != nil {
75+
return err
76+
}
77+
return opts.RunAtlas()
4678
}
4779

48-
logs, err := opts.PodmanClient.ContainerLogs(ctx, opts.LocalMongodHostname())
80+
return errors.New("atlas deployments are not supported")
81+
}
82+
83+
func (opts *DownloadOpts) RunAtlas() error {
84+
if err := opts.downloadLogFile(); err != nil {
85+
return err
86+
}
87+
defer opts.Fs.Remove(opts.Out) //nolint:errcheck
88+
89+
output, err := opts.unzipFile()
4990
if err != nil {
5091
return err
5192
}
5293

94+
return opts.Print(output)
95+
}
96+
97+
func (opts *DownloadOpts) downloadLogFile() error {
98+
f, err := opts.NewWriteCloser()
99+
if err != nil {
100+
return err
101+
}
102+
defer f.Close()
103+
104+
r := opts.newHostLogsParams()
105+
if err = opts.downloadStore.DownloadLog(f, r); err != nil {
106+
_ = opts.OnError(f)
107+
}
108+
return err
109+
}
110+
111+
func (opts *DownloadOpts) unzipFile() (string, error) {
112+
gzippedFile, err := opts.Fs.OpenFile(opts.Out, os.O_RDONLY, filePermission)
113+
if err != nil {
114+
return "", err
115+
}
116+
defer gzippedFile.Close()
117+
118+
gzReader, err := gzip.NewReader(gzippedFile)
119+
if err != nil {
120+
return "", err
121+
}
122+
defer gzReader.Close()
123+
124+
s, err := io.ReadAll(gzReader)
125+
if err != nil {
126+
return "", err
127+
}
128+
129+
return string(s), nil
130+
}
131+
132+
func (opts *DownloadOpts) newHostLogsParams() *admin.GetHostLogsApiParams {
133+
fileBaseName := strings.TrimSuffix(opts.name, filepath.Ext(opts.name))
134+
params := &admin.GetHostLogsApiParams{
135+
GroupId: opts.ConfigProjectID(),
136+
HostName: opts.host,
137+
LogName: fileBaseName,
138+
}
139+
if opts.start > 0 {
140+
params.StartDate = &opts.start
141+
}
142+
if opts.end > 0 {
143+
params.StartDate = &opts.end
144+
}
145+
return params
146+
}
147+
148+
func (opts *DownloadOpts) RunLocal(ctx context.Context) error {
149+
logs, err := opts.PodmanClient.ContainerLogs(ctx, opts.LocalMongodHostname())
150+
if err != nil {
151+
return err
152+
}
53153
// format log entries into lines
54154
if opts.IsJSONOutput() {
55155
return opts.Print(logs)
56156
}
57157
return opts.Print(strings.Join(logs, "\n"))
58158
}
59159

160+
func (opts *DownloadOpts) validateAtlasFlags() error {
161+
if opts.host == "" {
162+
return errors.New("missing --hostname flag")
163+
}
164+
if opts.name == "" {
165+
return errors.New("missing --name flag")
166+
}
167+
168+
validNameFlags := []string{"mongodb.gz", "mongos.gz", "mongosqld.gz", "mongodb-audit-log.gz", "mongos-audit-log.gz"}
169+
if !search.StringInSliceFold(validNameFlags, opts.name) {
170+
return fmt.Errorf("invalid --name flag: %s", opts.name)
171+
}
172+
return nil
173+
}
174+
60175
// atlas deployments logs.
61176
func LogsBuilder() *cobra.Command {
62177
opts := &DownloadOpts{}
@@ -68,19 +183,30 @@ func LogsBuilder() *cobra.Command {
68183
GroupID: "local",
69184
PreRunE: func(cmd *cobra.Command, args []string) error {
70185
w := cmd.OutOrStdout()
71-
log.SetWriter(w)
186+
opts.Fs = afero.NewOsFs()
187+
opts.Out = opts.name
72188

73189
return opts.PreRunE(
74190
opts.InitStore(cmd.Context(), cmd.OutOrStdout()),
191+
opts.initStore(cmd.Context()),
75192
opts.InitOutput(w, ""))
76193
},
77194
RunE: func(cmd *cobra.Command, args []string) error {
78195
return opts.Run(cmd.Context())
79196
},
80197
}
81198

199+
cmd.Flags().StringVar(&opts.DeploymentType, flag.TypeFlag, "", usage.DeploymentType)
82200
cmd.Flags().StringVarP(&opts.Output, flag.Output, flag.OutputShort, "", usage.FormatOut)
83201
cmd.Flags().StringVar(&opts.DeploymentName, flag.DeploymentName, "", usage.DeploymentName)
84202

203+
// Atlas flags
204+
cmd.Flags().Int64Var(&opts.start, flag.Start, 0, usage.LogStart)
205+
cmd.Flags().Int64Var(&opts.end, flag.End, 0, usage.LogEnd)
206+
cmd.Flags().BoolVar(&opts.Force, flag.Force, false, usage.ForceFile)
207+
cmd.Flags().StringVar(&opts.ProjectID, flag.ProjectID, "", usage.ProjectID)
208+
cmd.Flags().StringVar(&opts.host, flag.Hostname, "", usage.LogHostName)
209+
cmd.Flags().StringVar(&opts.name, flag.Name, "", usage.LogName)
210+
85211
return cmd
86212
}

internal/cli/atlas/deployments/logs_test.go

+55-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,10 @@ package deployments
1818

1919
import (
2020
"bytes"
21+
"compress/gzip"
2122
"context"
23+
"io"
24+
"os"
2225
"strings"
2326
"testing"
2427

@@ -27,7 +30,10 @@ import (
2730
"github.com/mongodb/mongodb-atlas-cli/internal/cli/atlas/deployments/options"
2831
"github.com/mongodb/mongodb-atlas-cli/internal/cli/atlas/deployments/test/fixture"
2932
"github.com/mongodb/mongodb-atlas-cli/internal/flag"
33+
"github.com/mongodb/mongodb-atlas-cli/internal/mocks"
3034
"github.com/mongodb/mongodb-atlas-cli/internal/test"
35+
"github.com/spf13/afero"
36+
"go.mongodb.org/atlas-sdk/v20230201008/admin"
3137
)
3238

3339
func TestLogsBuilder(t *testing.T) {
@@ -46,7 +52,7 @@ func TestLogs_RunLocal(t *testing.T) {
4652
ctrl := gomock.NewController(t)
4753
ctx := context.Background()
4854
buf := new(bytes.Buffer)
49-
expectedLocalDeployment := "localDeployment1"
55+
expectedLocalDeployment := "localDeployment"
5056
deploymentTest := fixture.NewMockLocalDeploymentOpts(ctrl, expectedLocalDeployment)
5157
mockPodman := deploymentTest.MockPodman
5258

@@ -75,3 +81,51 @@ func TestLogs_RunLocal(t *testing.T) {
7581
t.Fatalf("Run() expected output: %s, got: %s", expectedLogs, buf.String())
7682
}
7783
}
84+
85+
func TestLogs_RunAtlas(t *testing.T) {
86+
ctrl := gomock.NewController(t)
87+
ctx := context.Background()
88+
buf := new(bytes.Buffer)
89+
atlasDeployment := "localDeployment1"
90+
mockStore := mocks.NewMockLogsDownloader(ctrl)
91+
deploymentTest := fixture.NewMockAtlasDeploymentOpts(ctrl, atlasDeployment)
92+
93+
downloadOpts := &DownloadOpts{
94+
OutputOpts: cli.OutputOpts{OutWriter: buf},
95+
GlobalOpts: cli.GlobalOpts{ProjectID: "ProjectID"},
96+
DownloaderOpts: cli.DownloaderOpts{},
97+
DeploymentOpts: *deploymentTest.Opts,
98+
downloadStore: mockStore,
99+
host: "test",
100+
name: "mongodb.gz",
101+
}
102+
103+
downloadOpts.Fs = afero.NewMemMapFs()
104+
downloadOpts.Out = downloadOpts.name
105+
deploymentTest.CommonAtlasMocks(downloadOpts.ProjectID)
106+
107+
mockStore.
108+
EXPECT().
109+
DownloadLog(gomock.Any(), downloadOpts.newHostLogsParams()).
110+
Times(1).
111+
DoAndReturn(func(_ io.Writer, _ *admin.GetHostLogsApiParams) error {
112+
f, err := downloadOpts.Fs.OpenFile(downloadOpts.Out, os.O_WRONLY|os.O_CREATE, 0644)
113+
if err != nil {
114+
return err
115+
}
116+
117+
zw := gzip.NewWriter(f)
118+
defer zw.Close()
119+
zw.Name = downloadOpts.Out
120+
121+
if _, err = zw.Write([]byte("log\nlog\n")); err != nil {
122+
return err
123+
}
124+
125+
return nil
126+
})
127+
128+
if err := downloadOpts.Run(ctx); err != nil {
129+
t.Fatalf("Run() unexpected error: %v", err)
130+
}
131+
}

internal/cli/atlas/logs/download.go

+3-1
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,7 @@ func (opts *DownloadOpts) Run() error {
5757
if err != nil {
5858
return err
5959
}
60+
defer f.Close()
6061

6162
r := opts.newHostLogsParams()
6263
if err := opts.store.DownloadLog(f, r); err != nil {
@@ -66,7 +67,8 @@ func (opts *DownloadOpts) Run() error {
6667
if !opts.ShouldDownloadToStdout() {
6768
fmt.Printf(downloadMessage, opts.Out)
6869
}
69-
return f.Close()
70+
71+
return nil
7072
}
7173

7274
func (opts *DownloadOpts) initDefaultOut() error {

internal/usage/usage.go

+2
Original file line numberDiff line numberDiff line change
@@ -465,4 +465,6 @@ dbName and collection are required only for built-in roles.`
465465
ConnectWith = "Method for connecting to the deployment. Valid values are mongosh, compass, and skip."
466466
AlertConfigFilename = "Path to the JSON configuration file that defines alert configuration settings."
467467
DeploymentName = "Name of the deployment."
468+
LogName = "Name of the log file (e.g. mongodb.gz|mongos.gz|mongosqld.gz|mongodb-audit-log.gz|mongos-audit-log.gz)."
469+
LogHostName = "Name of the host that stores the log files that you want to download."
468470
)

0 commit comments

Comments
 (0)