-
Notifications
You must be signed in to change notification settings - Fork 3
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
Fix: Skip s3 upload if record's last-modified time matches DB value #921
Changes from 7 commits
9741095
110efe1
4110e9a
e4dccd7
cf0c744
5f85a6d
37c4b8b
d22526e
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 |
---|---|---|
|
@@ -17,14 +17,12 @@ import ( | |
goenv "github.com/Netflix/go-env" | ||
"github.com/aws/aws-lambda-go/events" | ||
"github.com/aws/aws-sdk-go-v2/aws" | ||
awsTransport "github.com/aws/aws-sdk-go-v2/aws/transport/http" | ||
"github.com/aws/aws-sdk-go-v2/config" | ||
"github.com/aws/aws-sdk-go-v2/credentials" | ||
"github.com/aws/aws-sdk-go-v2/feature/dynamodb/attributevalue" | ||
"github.com/aws/aws-sdk-go-v2/service/dynamodb" | ||
ddbtypes "github.com/aws/aws-sdk-go-v2/service/dynamodb/types" | ||
"github.com/aws/aws-sdk-go-v2/service/s3" | ||
smithyhttp "github.com/aws/smithy-go/transport/http" | ||
"github.com/go-kit/log" | ||
"github.com/hashicorp/go-multierror" | ||
"github.com/johannesboyne/gofakes3" | ||
|
@@ -53,9 +51,8 @@ func (m mockDDBClientGetItemCollection) NewGetItemClient(t *testing.T) mockDynam | |
err := attributevalue.UnmarshalMap(params.Key, &getItemKey) | ||
require.NoError(t, err, "Failed to extract grant_id value from DynamoDB GetItem key") | ||
output := dynamodb.GetItemOutput{Item: nil} | ||
targetGrantId, exists := getItemKey["grant_id"] | ||
var rvErr error | ||
if exists { | ||
if targetGrantId, exists := getItemKey["grant_id"]; exists { | ||
Comment on lines
-56
to
+55
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. This is just a change in variable grouping that I noticed as a readability issue; it has nothing to do with the rest of the PR. |
||
for _, rv := range m { | ||
if rv.GrantId == targetGrantId { | ||
output.Item = map[string]ddbtypes.AttributeValue{ | ||
|
@@ -602,54 +599,46 @@ func TestProcessRecord(t *testing.T) { | |
|
||
t.Run("Error getting item from DynamoDB", func(t *testing.T) { | ||
setupLambdaEnvForTesting(t) | ||
c := mockS3ReadwriteObjectAPI{ | ||
mockHeadObjectAPI( | ||
func(ctx context.Context, params *s3.HeadObjectInput, optFns ...func(*s3.Options)) (*s3.HeadObjectOutput, error) { | ||
t.Helper() | ||
return &s3.HeadObjectOutput{}, fmt.Errorf("server error") | ||
}, | ||
), | ||
mockGetObjectAPI(nil), | ||
mockPutObjectAPI(nil), | ||
} | ||
s3client := mockPutObjectAPI(func(context.Context, *s3.PutObjectInput, ...func(*s3.Options)) (*s3.PutObjectOutput, error) { | ||
t.Helper() | ||
require.Fail(t, "PutObject called unexpectedly") | ||
return nil, nil | ||
}) | ||
ddbLookups := make(mockDDBClientGetItemCollection, 0) | ||
ddbLookups = append(ddbLookups, mockDDBClientGetItemReturnValue{ | ||
GrantId: string(testOpportunity.OpportunityID), | ||
ItemLastModified: string(testOpportunity.LastUpdatedDate), | ||
GetItemErr: errors.New("Some issue with DynamoDB"), | ||
}) | ||
err := processRecord(context.TODO(), c, ddbLookups.NewGetItemClient(t), testOpportunity) | ||
err := processRecord(context.TODO(), s3client, ddbLookups.NewGetItemClient(t), testOpportunity) | ||
Comment on lines
-605
to
+613
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. This change corresponds to the simplified |
||
assert.ErrorContains(t, err, "Error determining last modified time for remote record") | ||
}) | ||
|
||
t.Run("Error uploading to S3", func(t *testing.T) { | ||
setupLambdaEnvForTesting(t) | ||
s3Client := mockS3ReadwriteObjectAPI{ | ||
mockHeadObjectAPI( | ||
func(context.Context, *s3.HeadObjectInput, ...func(*s3.Options)) (*s3.HeadObjectOutput, error) { | ||
t.Helper() | ||
return nil, &awsTransport.ResponseError{ | ||
ResponseError: &smithyhttp.ResponseError{Response: &smithyhttp.Response{ | ||
Response: &http.Response{StatusCode: 404}, | ||
}}, | ||
} | ||
}, | ||
), | ||
mockGetObjectAPI(func(context.Context, *s3.GetObjectInput, ...func(*s3.Options)) (*s3.GetObjectOutput, error) { | ||
t.Helper() | ||
require.Fail(t, "GetObject called unexpectedly") | ||
return nil, nil | ||
}), | ||
mockPutObjectAPI(func(context.Context, *s3.PutObjectInput, ...func(*s3.Options)) (*s3.PutObjectOutput, error) { | ||
t.Helper() | ||
return nil, fmt.Errorf("some PutObject error") | ||
}), | ||
} | ||
fmt.Printf("%T", s3Client) | ||
s3Client := mockPutObjectAPI(func(context.Context, *s3.PutObjectInput, ...func(*s3.Options)) (*s3.PutObjectOutput, error) { | ||
t.Helper() | ||
return nil, fmt.Errorf("some PutObject error") | ||
}) | ||
Comment on lines
-627
to
+622
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. This change corresponds to the simplified |
||
ddb := mockDDBClientGetItemCollection([]mockDDBClientGetItemReturnValue{ | ||
{GrantId: string(testOpportunity.OpportunityID), ItemLastModified: string(testOpportunity.LastUpdatedDate)}, | ||
// Do not provide a matching record, ensuring that processRecord() will attempt to upload | ||
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. Fixes a test that started to fail once the date-time equality check in |
||
}) | ||
err := processRecord(context.TODO(), s3Client, ddb.NewGetItemClient(t), testOpportunity) | ||
assert.ErrorContains(t, err, "Error uploading prepared grant record to S3") | ||
}) | ||
|
||
t.Run("matching LastUpdatedDate skips upload to S3", func(t *testing.T) { | ||
setupLambdaEnvForTesting(t) | ||
s3Client := mockPutObjectAPI(func(context.Context, *s3.PutObjectInput, ...func(*s3.Options)) (*s3.PutObjectOutput, error) { | ||
t.Helper() | ||
require.Fail(t, "PutObject called unexpectedly") | ||
return nil, fmt.Errorf("PutObject called unexpectedly") | ||
}) | ||
ddb := mockDDBClientGetItemCollection([]mockDDBClientGetItemReturnValue{{ | ||
GrantId: string(testOpportunity.OpportunityID), | ||
ItemLastModified: string(testOpportunity.LastUpdatedDate), | ||
}}) | ||
err := processRecord(context.TODO(), s3Client, ddb.NewGetItemClient(t), testOpportunity) | ||
assert.NoError(t, err) | ||
}) | ||
Comment on lines
+630
to
+643
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. Covers the modified date-time equality check in |
||
} |
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. Some of the mock interfaces and structs in this file are no longer needed since |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -73,8 +73,8 @@ func (f forecast) toXML() ([]byte, error) { | |
return xml.Marshal(grantsgov.OpportunityForecastDetail_1_0(f)) | ||
} | ||
|
||
func (o forecast) dynamoDBItemKey() map[string]ddbtypes.AttributeValue { | ||
func (f forecast) dynamoDBItemKey() map[string]ddbtypes.AttributeValue { | ||
return map[string]ddbtypes.AttributeValue{ | ||
"grant_id": &ddbtypes.AttributeValueMemberS{Value: string(o.OpportunityID)}, | ||
"grant_id": &ddbtypes.AttributeValueMemberS{Value: string(f.OpportunityID)}, | ||
Comment on lines
+76
to
+78
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. Just fixing a variable name that I noticed; likely due to a copy/paste flub. |
||
} | ||
} |
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. Note that the dashboard still counts metrics named according to the old (pre-#848) |
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. Updates the Datadog metric definitions according to the new terminology. |
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.
This is a simplified interface since we don't need to perform read operations anymore.