-
Notifications
You must be signed in to change notification settings - Fork 9
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add ConfigureIndex method #34
Changes from 12 commits
ce920dc
1b6c867
366b4bc
1b58b5f
89e3368
7fb2f0c
1b49970
018feb0
21ce7d6
a99a56f
75713e5
70e0bf2
0536f5c
cb6b716
f53f536
adc7e3a
f3939ed
a3049bd
1046619
6769041
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -273,6 +273,104 @@ func (c *Client) DeleteIndex(ctx context.Context, idxName string) error { | |||||||||||||||||||||||||
return nil | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
// ConfigureIndex is used to [scale a pods-based index] up or down by changing the size of the pods or the number of | ||||||||||||||||||||||||||
//replicas. | ||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||
// Parameters: | ||||||||||||||||||||||||||
// - name: The name of the index to configure. | ||||||||||||||||||||||||||
// - pods: (Optional) The pod size to scale the index to (e.g. for a "p1" pod type, | ||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I like this formatting for There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yeah, sweet! |
||||||||||||||||||||||||||
// you could pass "p1.x2" to scale your index to the "x2" size, | ||||||||||||||||||||||||||
// or you could pass "p1.x4" to scale your index to the "x4" size, and | ||||||||||||||||||||||||||
// so forth. | ||||||||||||||||||||||||||
// - replicas: (Optional) The number of replicas to scale the index to. | ||||||||||||||||||||||||||
// This is capped by the maximum number of replicas allowed in your Pinecone project. To configure this number, | ||||||||||||||||||||||||||
// go to [app.pinecone.io], select your project, and configure the maximum number of pods. | ||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||
// Note: You can only scale an index up, not down. If you want to scale an index down, | ||||||||||||||||||||||||||
// you must create a new index with the desired configuration. | ||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||
// Returns a ConfigureIndexResponse object, which contains the new configuration of the index, or an error. | ||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||
// Example for a pods-based index originally configured with 1 "p1" pod of size "x2" and 1 replica: | ||||||||||||||||||||||||||
// // To scale the size of your pods from "x2" to "x4": | ||||||||||||||||||||||||||
// _, err := pc.ConfigureIndex(ctx, "my-index", "p1.x4", nil) | ||||||||||||||||||||||||||
// if err != nil { | ||||||||||||||||||||||||||
// fmt.Printf("Failed to configure index: %v\n", err) | ||||||||||||||||||||||||||
// } | ||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||
// // To scale the number of replicas: | ||||||||||||||||||||||||||
// _, err := pc.ConfigureIndex(ctx, "my-index", nil, 4) | ||||||||||||||||||||||||||
// if err != nil { | ||||||||||||||||||||||||||
// fmt.Printf("Failed to configure index: %v\n", err) | ||||||||||||||||||||||||||
// } | ||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||
// // To scale both the size of your pods and the number of replicas: | ||||||||||||||||||||||||||
// _, err := pc.ConfigureIndex(ctx, "my-index", "p1.x4", 4) | ||||||||||||||||||||||||||
// if err != nil { | ||||||||||||||||||||||||||
// fmt.Printf("Failed to configure index: %v\n", err) | ||||||||||||||||||||||||||
// } | ||||||||||||||||||||||||||
// | ||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Did you happen to run I think There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ah I didn't -- this was pushed before before our Slack convo about There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Okay done :) |
||||||||||||||||||||||||||
// [scale a pods-based index]: https://docs.pinecone.io/guides/indexes/configure-pod-based-indexes | ||||||||||||||||||||||||||
// [app.pinecone.io]: https://app.pinecone.io | ||||||||||||||||||||||||||
func (c *Client) ConfigureIndex(ctx context.Context, name string, pods *string, | ||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Still, we use There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Okay, np, I'll change to |
||||||||||||||||||||||||||
replicas *int32) (*control.ConfigureIndexResponse, | ||||||||||||||||||||||||||
error) { | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
var podType *control.PodSpecPodType | ||||||||||||||||||||||||||
var replicasAmt *control.PodSpecReplicas | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
if pods == nil && replicas == nil { | ||||||||||||||||||||||||||
return nil, fmt.Errorf("Must specify either pods or replicas") | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
if pods != nil { | ||||||||||||||||||||||||||
podTypeVal := control.PodSpecPodType(*pods) | ||||||||||||||||||||||||||
podType = &podTypeVal | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
if replicas != nil { | ||||||||||||||||||||||||||
replicasVal := control.PodSpecReplicas(*replicas) | ||||||||||||||||||||||||||
replicasAmt = &replicasVal | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
request := control.ConfigureIndexRequest{ | ||||||||||||||||||||||||||
Spec: struct { | ||||||||||||||||||||||||||
Pod struct { | ||||||||||||||||||||||||||
PodType *control.PodSpecPodType `json:"pod_type,omitempty"` | ||||||||||||||||||||||||||
Replicas *control.PodSpecReplicas `json:"replicas,omitempty"` | ||||||||||||||||||||||||||
} `json:"pod"` | ||||||||||||||||||||||||||
}{ | ||||||||||||||||||||||||||
Pod: struct { | ||||||||||||||||||||||||||
PodType *control.PodSpecPodType `json:"pod_type,omitempty"` | ||||||||||||||||||||||||||
Replicas *control.PodSpecReplicas `json:"replicas,omitempty"` | ||||||||||||||||||||||||||
}{ | ||||||||||||||||||||||||||
PodType: podType, | ||||||||||||||||||||||||||
Replicas: replicasAmt, | ||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||
}, | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
req, err := c.restClient.ConfigureIndex(ctx, name, request) | ||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can we rename There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Sure! |
||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||
log.Fatalf("Failed to configure index %s. Error: %v", name, err) | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
if req.StatusCode != http.StatusOK { | ||||||||||||||||||||||||||
err := handleErrorResponseBody(req, "failed to configure index: ") | ||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||
return nil, err | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think here you can just do:
Suggested change
We can assume if the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. oo nice! |
||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
response, err := control.ParseConfigureIndexResponse(req) // TODO: need this? | ||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't think we should be using the internals, I don't think we're doing that anywhere else. I'd maybe look at how we're handling things in something like
Suggested change
Ultimately this just returns an There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Sure yeah good call, I'll do that! |
||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||
log.Fatalf("Failed to configure index %s. Error: %v", name, err) | ||||||||||||||||||||||||||
return nil, err | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
defer req.Body.Close() | ||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nit: In other functions we put this line right after the function call so it's a bit more clear we're actually closing it, so I'd maybe move it up to after the initial function call and error check:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. okay will do! |
||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
return response, nil | ||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||
|
||||||||||||||||||||||||||
func (c *Client) ListCollections(ctx context.Context) ([]*Collection, error) { | ||||||||||||||||||||||||||
res, err := c.restClient.ListCollections(ctx) | ||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -3,18 +3,18 @@ package pinecone | |
import ( | ||
"context" | ||
"fmt" | ||
"github.com/google/uuid" | ||
"github.com/pinecone-io/go-pinecone/internal/mocks" | ||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/require" | ||
"github.com/stretchr/testify/suite" | ||
"io" | ||
"log" | ||
"net/http" | ||
"os" | ||
"reflect" | ||
"strings" | ||
"testing" | ||
|
||
"github.com/google/uuid" | ||
"github.com/pinecone-io/go-pinecone/internal/mocks" | ||
"github.com/stretchr/testify/assert" | ||
"github.com/stretchr/testify/require" | ||
"github.com/stretchr/testify/suite" | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nit: I think There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I thinkkk so. This was also pre-our formatting convo, so I'll push up some changes wrt that soon! |
||
) | ||
|
||
type ClientTests struct { | ||
|
@@ -496,6 +496,126 @@ func (ts *ClientTests) TestDeleteCollection() { | |
require.NoError(ts.T(), err) | ||
} | ||
|
||
func (ts *ClientTests) TestConfigureIndexIllegalScaleDown() { | ||
name := uuid.New().String() | ||
|
||
defer func(ts *ClientTests, name string) { | ||
err := ts.deleteIndex(name) | ||
require.NoError(ts.T(), err) | ||
}(ts, name) | ||
|
||
_, err := ts.client.CreatePodIndex(context.Background(), &CreatePodIndexRequest{ | ||
Name: name, | ||
Dimension: 10, | ||
Metric: Cosine, | ||
Environment: "us-east1-gcp", | ||
PodType: "p1.x2", | ||
},) | ||
if err != nil { | ||
log.Fatalf("Error creating index %s: %v", name, err) | ||
} | ||
|
||
pods := "p1.x1" // test index originally created with "p1.x2" pods | ||
replicas := int32(1) // could be nil, but do not want to test nil case here | ||
_, err = ts.client.ConfigureIndex(context.Background(), name, &pods, &replicas) | ||
require.ErrorContainsf(ts.T(), err, "Cannot scale down", err.Error()) | ||
} | ||
|
||
func (ts *ClientTests) TestConfigureIndexScaleUpNoPods() { | ||
name := uuid.New().String() | ||
|
||
defer func(ts *ClientTests, name string) { | ||
err := ts.deleteIndex(name) | ||
require.NoError(ts.T(), err) | ||
}(ts, name) | ||
|
||
_, err := ts.client.CreatePodIndex(context.Background(), &CreatePodIndexRequest{ | ||
Name: name, | ||
Dimension: 10, | ||
Metric: Cosine, | ||
Environment: "us-east1-gcp", | ||
PodType: "p1.x2", | ||
},) | ||
if err != nil { | ||
log.Fatalf("Error creating index %s: %v", name, err) | ||
} | ||
|
||
replicas := int32(2) | ||
_, err = ts.client.ConfigureIndex(context.Background(), name, nil, &replicas) | ||
require.NoError(ts.T(), err) | ||
} | ||
|
||
func (ts *ClientTests) TestConfigureIndexScaleUpNoReplicas() { | ||
name := uuid.New().String() | ||
|
||
defer func(ts *ClientTests, name string) { | ||
err := ts.deleteIndex(name) | ||
require.NoError(ts.T(), err) | ||
}(ts, name) | ||
|
||
_, err := ts.client.CreatePodIndex(context.Background(), &CreatePodIndexRequest{ | ||
Name: name, | ||
Dimension: 10, | ||
Metric: Cosine, | ||
Environment: "us-east1-gcp", | ||
PodType: "p1.x2", | ||
},) | ||
if err != nil { | ||
log.Fatalf("Error creating index %s: %v", name, err) | ||
} | ||
|
||
pods := "p1.x4" | ||
_, err = ts.client.ConfigureIndex(context.Background(), name, &pods, nil) | ||
require.NoError(ts.T(), err) | ||
} | ||
|
||
func (ts *ClientTests) TestConfigureIndexIllegalNoPodsOrReplicas() { | ||
name := uuid.New().String() | ||
|
||
defer func(ts *ClientTests, name string) { | ||
err := ts.deleteIndex(name) | ||
require.NoError(ts.T(), err) | ||
}(ts, name) | ||
|
||
_, err := ts.client.CreatePodIndex(context.Background(), &CreatePodIndexRequest{ | ||
Name: name, | ||
Dimension: 10, | ||
Metric: Cosine, | ||
Environment: "us-east1-gcp", | ||
PodType: "p1.x2", | ||
},) | ||
if err != nil { | ||
log.Fatalf("Error creating index %s: %v", name, err) | ||
} | ||
|
||
_, err = ts.client.ConfigureIndex(context.Background(), name, nil, nil) | ||
require.ErrorContainsf(ts.T(), err, "Must specify either pods or replicas", err.Error()) | ||
} | ||
|
||
func (ts *ClientTests) TestConfigureIndexHitPodLimit() { | ||
name := uuid.New().String() | ||
|
||
defer func(ts *ClientTests, name string) { | ||
err := ts.deleteIndex(name) | ||
require.NoError(ts.T(), err) | ||
}(ts, name) | ||
|
||
_, err := ts.client.CreatePodIndex(context.Background(), &CreatePodIndexRequest{ | ||
Name: name, | ||
Dimension: 10, | ||
Metric: Cosine, | ||
Environment: "us-east1-gcp", | ||
PodType: "p1.x2", | ||
},) | ||
if err != nil { | ||
log.Fatalf("Error creating index %s: %v", name, err) | ||
} | ||
|
||
replicas := int32(30000) | ||
_, err = ts.client.ConfigureIndex(context.Background(), name, nil, &replicas) | ||
require.ErrorContainsf(ts.T(), err, "You've reached the max pods allowed", err.Error()) | ||
} | ||
|
||
func (ts *ClientTests) deleteIndex(name string) error { | ||
return ts.client.DeleteIndex(context.Background(), name) | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like a space is needed between
//
andreplicas
.