Skip to content

Commit 17598c6

Browse files
CLOUDP-164347: Add --flag file to the atlas cluster index create command (#2768)
1 parent 1181f26 commit 17598c6

21 files changed

+254
-22
lines changed

docs/command/atlas-clusters-indexes-create.txt

+21-3
Original file line numberDiff line numberDiff line change
@@ -59,20 +59,32 @@ Options
5959
- Name of the cluster.
6060
* - --collection
6161
- string
62-
- true
62+
- false
6363
- Name of the collection.
64+
65+
Mutually exclusive with --file.
6466
* - --db
6567
- string
66-
- true
68+
- false
6769
- Name of the database.
70+
71+
Mutually exclusive with --file.
72+
* - -f, --file
73+
- string
74+
- false
75+
- Path to an optional JSON configuration file that defines index settings.
76+
77+
Mutually exclusive with --db, --collection, --key.
6878
* - -h, --help
6979
-
7080
- false
7181
- help for create
7282
* - --key
7383
- strings
74-
- true
84+
- false
7585
- Field to be indexed and the type of index in the following format: field:type.
86+
87+
Mutually exclusive with --file.
7688
* - --projectId
7789
- string
7890
- false
@@ -112,3 +124,9 @@ Examples
112124
# Create a compound index named property_room_bedrooms on the
113125
listings collection of the realestate database:
114126
atlas clusters indexes create property_room_bedrooms --clusterName Cluster0 --collection listings --db realestate --key property_type:1 --key room_type:1 --key bedrooms:1
127+
128+
129+
.. code-block::
130+
131+
# Create an index named my_index from a JSON configuration file named myfile.json:
132+
atlas clusters indexes create my_index --clusterName Cluster0 --file file.json

internal/cli/atlas/clusters/indexes/create.go

+56-8
Original file line numberDiff line numberDiff line change
@@ -22,9 +22,11 @@ import (
2222
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/cli"
2323
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/cli/require"
2424
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/config"
25+
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/file"
2526
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/flag"
2627
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/store"
2728
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/usage"
29+
"github.com/spf13/afero"
2830
"github.com/spf13/cobra"
2931
atlasv2 "go.mongodb.org/atlas-sdk/v20231115007/admin"
3032
)
@@ -35,9 +37,11 @@ type CreateOpts struct {
3537
name string
3638
db string
3739
collection string
40+
filename string
3841
keys []string
3942
sparse bool
4043
store store.IndexCreator
44+
fs afero.Fs
4145
}
4246

4347
func (opts *CreateOpts) initStore(ctx context.Context) func() error {
@@ -61,11 +65,43 @@ func (opts *CreateOpts) Run() error {
6165
}
6266

6367
func (opts *CreateOpts) newIndex() (*atlasv2.DatabaseRollingIndexRequest, error) {
68+
if opts.filename != "" {
69+
return opts.newIndexViaFile()
70+
}
71+
72+
return opts.newIndexViaFlags()
73+
}
74+
75+
func (opts *CreateOpts) newIndexViaFile() (*atlasv2.DatabaseRollingIndexRequest, error) {
76+
i := new(atlasv2.DatabaseRollingIndexRequest)
77+
if err := file.Load(opts.fs, opts.filename, i); err != nil {
78+
return nil, err
79+
}
80+
81+
if opts.name == "" {
82+
return i, nil
83+
}
84+
85+
if i.Options == nil {
86+
i.Options = &atlasv2.IndexOptions{
87+
Name: &opts.name,
88+
}
89+
90+
return i, nil
91+
}
92+
93+
i.Options.Name = &opts.name
94+
95+
return i, nil
96+
}
97+
98+
func (opts *CreateOpts) newIndexViaFlags() (*atlasv2.DatabaseRollingIndexRequest, error) {
99+
i := new(atlasv2.DatabaseRollingIndexRequest)
64100
keys, err := opts.indexKeys()
65101
if err != nil {
66102
return nil, err
67103
}
68-
i := new(atlasv2.DatabaseRollingIndexRequest)
104+
69105
i.Db = opts.db
70106
i.Collection = opts.collection
71107
i.Keys = &keys
@@ -100,9 +136,11 @@ func (opts *CreateOpts) indexKeys() ([]map[string]string, error) {
100136
}
101137

102138
// CreateBuilder builds a cobra.Command that can run as:
103-
// mcli atlas clusters index create [indexName] --clusterName clusterName --collection collection --dbName dbName [--key field:type].
139+
// mcli atlas clusters index create [indexName] --clusterName clusterName --collection collection --dbName dbName [--key field:type] --file filename.
104140
func CreateBuilder() *cobra.Command {
105-
opts := &CreateOpts{}
141+
opts := &CreateOpts{
142+
fs: afero.NewOsFs(),
143+
}
106144
cmd := &cobra.Command{
107145
Use: "create [indexName]",
108146
Short: "Create a rolling index for the specified cluster for your project.",
@@ -116,8 +154,17 @@ func CreateBuilder() *cobra.Command {
116154
117155
# Create a compound index named property_room_bedrooms on the
118156
listings collection of the realestate database:
119-
atlas clusters indexes create property_room_bedrooms --clusterName Cluster0 --collection listings --db realestate --key property_type:1 --key room_type:1 --key bedrooms:1`,
157+
atlas clusters indexes create property_room_bedrooms --clusterName Cluster0 --collection listings --db realestate --key property_type:1 --key room_type:1 --key bedrooms:1
158+
159+
# Create an index named my_index from a JSON configuration file named myfile.json:
160+
atlas clusters indexes create my_index --clusterName Cluster0 --file file.json`,
120161
PreRunE: func(cmd *cobra.Command, _ []string) error {
162+
if opts.filename == "" {
163+
_ = cmd.MarkFlagRequired(flag.Database)
164+
_ = cmd.MarkFlagRequired(flag.Collection)
165+
_ = cmd.MarkFlagRequired(flag.Key)
166+
}
167+
121168
return opts.PreRunE(opts.ValidateProjectID, opts.initStore(cmd.Context()))
122169
},
123170
RunE: func(_ *cobra.Command, args []string) error {
@@ -133,13 +180,14 @@ func CreateBuilder() *cobra.Command {
133180
cmd.Flags().StringVar(&opts.collection, flag.Collection, "", usage.Collection)
134181
cmd.Flags().StringSliceVar(&opts.keys, flag.Key, []string{}, usage.Key)
135182
cmd.Flags().BoolVar(&opts.sparse, flag.Sparse, false, usage.Sparse)
183+
cmd.Flags().StringVarP(&opts.filename, flag.File, flag.FileShort, "", usage.IndexFilename)
136184

137185
cmd.Flags().StringVar(&opts.ProjectID, flag.ProjectID, "", usage.ProjectID)
138186

139-
_ = cmd.MarkFlagRequired(flag.ClusterName)
140-
_ = cmd.MarkFlagRequired(flag.Database)
141-
_ = cmd.MarkFlagRequired(flag.Collection)
142-
_ = cmd.MarkFlagRequired(flag.Key)
187+
cmd.MarkFlagsMutuallyExclusive(flag.File, flag.Database)
188+
cmd.MarkFlagsMutuallyExclusive(flag.File, flag.Collection)
189+
cmd.MarkFlagsMutuallyExclusive(flag.File, flag.Key)
143190

191+
_ = cmd.MarkFlagRequired(flag.ClusterName)
144192
return cmd
145193
}

internal/cli/atlas/clusters/indexes/create_test.go

+49-1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ import (
2323
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/flag"
2424
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/mocks"
2525
"github.com/mongodb/mongodb-atlas-cli/atlascli/internal/test"
26+
"github.com/spf13/afero"
2627
)
2728

2829
func TestCreate_Run(t *testing.T) {
@@ -50,11 +51,58 @@ func TestCreate_Run(t *testing.T) {
5051
}
5152
}
5253

54+
func TestCreateWithFile_Run(t *testing.T) {
55+
ctrl := gomock.NewController(t)
56+
mockStore := mocks.NewMockIndexCreator(ctrl)
57+
appFS := afero.NewMemMapFs()
58+
fileJSON := `
59+
{
60+
"collection": "collectionName",
61+
"db": "dbName",
62+
"options":{
63+
"sparse": true,
64+
"unique": true,
65+
"textIndexVersion": 1,
66+
"name": "myIndex",
67+
"min": 1,
68+
"max": 10,
69+
"language_override": "test",
70+
"hidden": true,
71+
"expireAfterSeconds": 2,
72+
"default_language": "test",
73+
"default_language": "test",
74+
"columnstoreProjection": {"key":1, "key2":2},
75+
"bucketSize": 2,
76+
"bits": 222,
77+
"background": false,
78+
"2dsphereIndexVersion": 2
79+
}
80+
}`
81+
fileName := "atlas_cluster_index_create_test.json"
82+
_ = afero.WriteFile(appFS, fileName, []byte(fileJSON), 0600)
83+
createOpts := &CreateOpts{
84+
filename: fileName,
85+
store: mockStore,
86+
fs: appFS,
87+
}
88+
89+
index, _ := createOpts.newIndex()
90+
mockStore.
91+
EXPECT().
92+
CreateIndex(createOpts.ProjectID, createOpts.clusterName, index).
93+
Return(nil).
94+
Times(1)
95+
96+
if err := createOpts.Run(); err != nil {
97+
t.Fatalf("Run() unexpected error: %v", err)
98+
}
99+
}
100+
53101
func TestCreateBuilder(t *testing.T) {
54102
test.CmdValidator(
55103
t,
56104
CreateBuilder(),
57105
0,
58-
[]string{flag.ClusterName, flag.Database, flag.Collection, flag.Key, flag.Sparse, flag.ProjectID},
106+
[]string{flag.ClusterName, flag.Database, flag.Collection, flag.Key, flag.Sparse, flag.ProjectID, flag.File},
59107
)
60108
}

internal/usage/usage.go

+1
Original file line numberDiff line numberDiff line change
@@ -107,6 +107,7 @@ dbName and collection are required only for built-in roles.`
107107
MaxDate = "Maximum created date. This option returns events whose created date is less than or equal to the specified value."
108108
MinDate = "Minimum created date. This option returns events whose created date is greater than or equal to the specified value."
109109
ClusterFilename = "Path to an optional JSON configuration file that defines cluster settings. To learn more about cluster configuration files for the Atlas CLI, see https://dochub.mongodb.org/core/cluster-config-file-atlascli. To learn more about cluster configuration files for MongoCLI, see https://dochub.mongodb.org/core/mms-cluster-settings-file-mcli."
110+
IndexFilename = "Path to an optional JSON configuration file that defines index settings."
110111
BackupFilename = "Path to an optional JSON configuration file that defines backup schedule settings. To learn about the cloud backup configuration file for the Atlas CLI, see https://dochub.mongodb.org/core/cloud-backup-config-file."
111112
SearchFilename = "Name of the JSON index configuration file to use. To learn about the Atlas Search index configuration file, see https://dochub.mongodb.org/core/search-index-config-file-atlascli. To learn about the Atlas Search index syntax and options that you can define in your configuration file, see https://dochub.mongodb.org/core/index-definitions-fts."
112113
SearchNodesFilename = "Name of the JSON index configuration file to use."

test/e2e/atlas/clusters_file_test.go

+46-4
Original file line numberDiff line numberDiff line change
@@ -83,12 +83,54 @@ func TestClustersFile(t *testing.T) {
8383
assert.Contains(t, string(resp), "Cluster available")
8484
})
8585

86+
t.Run("Create Partial Index", func(t *testing.T) {
87+
cmd := exec.Command(cliPath,
88+
clustersEntity,
89+
"indexes",
90+
"create",
91+
"--clusterName", clusterFileName,
92+
"--file=data/create_partial_index.json",
93+
"--projectId", g.projectID,
94+
)
95+
cmd.Env = os.Environ()
96+
resp, err := cmd.CombinedOutput()
97+
require.NoError(t, err, string(resp))
98+
})
99+
100+
t.Run("Create Sparse Index", func(t *testing.T) {
101+
cmd := exec.Command(cliPath,
102+
clustersEntity,
103+
"indexes",
104+
"create",
105+
"--clusterName", clusterFileName,
106+
"--file=data/create_sparse_index.json",
107+
"--projectId", g.projectID,
108+
)
109+
cmd.Env = os.Environ()
110+
resp, err := cmd.CombinedOutput()
111+
require.NoError(t, err, string(resp))
112+
})
113+
114+
t.Run("Create 2dspere Index", func(t *testing.T) {
115+
cmd := exec.Command(cliPath,
116+
clustersEntity,
117+
"indexes",
118+
"create",
119+
"--clusterName", clusterFileName,
120+
"--file=data/create_2dspere_index.json",
121+
"--projectId", g.projectID,
122+
)
123+
cmd.Env = os.Environ()
124+
resp, err := cmd.CombinedOutput()
125+
require.NoError(t, err, string(resp))
126+
})
127+
86128
t.Run("Update via file", func(t *testing.T) {
87129
cmd := exec.Command(cliPath,
88130
clustersEntity,
89131
"update",
90132
clusterFileName,
91-
"--file=update_cluster_test.json",
133+
"--file=data/update_cluster_test.json",
92134
"--projectId", g.projectID,
93135
"-o=json")
94136

@@ -141,9 +183,9 @@ func generateClusterFile(mdbVersion string) (string, error) {
141183
MongoDBMajorVersion: mdbVersion,
142184
}
143185

144-
templateFile := "create_cluster_test.json"
186+
templateFile := "data/create_cluster_test.json"
145187
if service := os.Getenv("MCLI_SERVICE"); service == config.CloudGovService {
146-
templateFile = "create_cluster_gov_test.json"
188+
templateFile = "data/create_cluster_gov_test.json"
147189
}
148190

149191
tmpl, err := template.ParseFiles(templateFile)
@@ -156,7 +198,7 @@ func generateClusterFile(mdbVersion string) (string, error) {
156198
return "", err
157199
}
158200

159-
const clusterFile = "create_cluster.json"
201+
const clusterFile = "data/create_cluster.json"
160202
file, err := os.Create(clusterFile)
161203
if err != nil {
162204
return "", err
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
{
2+
"collation": {
3+
"alternate": "non-ignorable",
4+
"backwards": false,
5+
"caseFirst": "lower",
6+
"caseLevel": false,
7+
"locale": "af",
8+
"maxVariable": "punct",
9+
"normalization": false,
10+
"numericOrdering": false,
11+
"strength": 3
12+
},
13+
"collection": "accounts",
14+
"db": "sample_airbnb",
15+
"keys": [
16+
{
17+
"test_field": "2dsphere"
18+
}
19+
],
20+
"options": {
21+
"name": "2dspereIndexTest",
22+
"2dsphereIndexVersion": 2
23+
}
24+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
{
2+
"collation": {
3+
"alternate": "non-ignorable",
4+
"backwards": false,
5+
"caseFirst": "lower",
6+
"caseLevel": false,
7+
"locale": "af",
8+
"maxVariable": "punct",
9+
"normalization": false,
10+
"numericOrdering": false,
11+
"strength": 3
12+
},
13+
"collection": "accounts",
14+
"db": "sample_airbnb",
15+
"keys": [
16+
{
17+
"property_type": "1",
18+
"room_type": "1"
19+
}
20+
],
21+
"options": {
22+
"name": "PartialIndexTest",
23+
"partialFilterExpression":{
24+
"limit": {"$gt": 900}
25+
}
26+
}
27+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
{
2+
"collation": {
3+
"alternate": "non-ignorable",
4+
"backwards": false,
5+
"caseFirst": "lower",
6+
"caseLevel": false,
7+
"locale": "af",
8+
"maxVariable": "punct",
9+
"normalization": false,
10+
"numericOrdering": false,
11+
"strength": 3
12+
},
13+
"collection": "accounts",
14+
"db": "sample_airbnb",
15+
"keys": [
16+
{
17+
"test_field": "1"
18+
}
19+
],
20+
"options": {
21+
"name": "SparseIndexTest",
22+
"sparse": true
23+
}
24+
}

0 commit comments

Comments
 (0)