Skip to content

Commit 2c13263

Browse files
authored
Add Batch Read support
* Implemented the `BatchGetItem` operation (#122). Co-authored-by: Juli Tera <terajul@amazon.com>
1 parent b16e7a5 commit 2c13263

13 files changed

+779
-102
lines changed

.github/workflows/ci.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -15,10 +15,10 @@ jobs:
1515
strategy:
1616
fail-fast: false
1717
matrix:
18-
ruby: [2.0, 2.1, 2.2, 2.3, 2.4, 2.5, 2.6, 2.7, '3.0', jruby-9.1, jruby-9.2]
18+
ruby: ['2.0', 2.1, 2.2, 2.3, 2.4, 2.5, 2.6, 2.7, '3.0', 3.1, 3.2, jruby-9.1, jruby-9.2, jruby-9.3, jruby-9.4]
1919
env: [NEW_RAILS, OLD_RAILS]
2020
exclude:
21-
- ruby: 2.0
21+
- ruby: '2.0'
2222
env: NEW_RAILS
2323
- ruby: 2.1
2424
env: NEW_RAILS

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
Unreleased Changes
22
------------------
33

4+
* Feature - Implement the `BatchGetItem` operation (#122)
5+
46
2.9.0 (2022-11-16)
57
------------------
68

README.md

+66
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,72 @@ item.active = false
8989
item.save
9090
```
9191

92+
### `BatchGetItem` and `BatchWriteItem`
93+
Aws Record provides [BatchGetItem](https://docs.aws.amazon.com/sdk-for-ruby/v3/api/Aws/DynamoDB/Client.html#batch_get_item-instance_method)
94+
and [BatchWriteItem](https://docs.aws.amazon.com/sdk-for-ruby/v3/api/Aws/DynamoDB/Client.html#batch_write_item-instance_method)
95+
support for aws-record models.
96+
97+
More info under the following documentation:
98+
* [Batch](https://docs.aws.amazon.com/sdk-for-ruby/aws-record/api/Aws/Record/Batch.html)
99+
* [BatchWrite](https://docs.aws.amazon.com/sdk-for-ruby/aws-record/api/Aws/Record/BatchWrite.html)
100+
* [BatchRead](https://docs.aws.amazon.com/sdk-for-ruby/aws-record/api/Aws/Record/BatchRead.html)
101+
102+
See examples below to see the feature in action.
103+
104+
**`BatchGetItem` Example:**
105+
```ruby
106+
class Lunch
107+
include Aws::Record
108+
integer_attr :id, hash_key: true
109+
string_attr :name, range_key: true
110+
end
111+
112+
class Dessert
113+
include Aws::Record
114+
integer_attr :id, hash_key: true
115+
string_attr :name, range_key: true
116+
end
117+
118+
# batch operations
119+
operation = Aws::Record::Batch.read do |db|
120+
db.find(Lunch, id: 1, name: 'Papaya Salad')
121+
db.find(Lunch, id: 2, name: 'BLT Sandwich')
122+
db.find(Dessert, id: 1, name: 'Apple Pie')
123+
end
124+
125+
# BatchRead is enumerable and handles pagination
126+
operation.each { |item| item.id }
127+
128+
# Alternatively, BatchRead provides a lower level interface through: execute!, complete? and items.
129+
# Unprocessed items can be processed by calling:
130+
operation.execute! until operation.complete?
131+
```
132+
133+
**`BatchWriteItem` Example:**
134+
```ruby
135+
class Breakfast
136+
include Aws::Record
137+
integer_attr :id, hash_key: true
138+
string_attr :name, range_key: true
139+
string_attr :body
140+
end
141+
142+
# setup
143+
eggs = Breakfast.new(id: 1, name: "eggs").save!
144+
waffles = Breakfast.new(id: 2, name: "waffles")
145+
pancakes = Breakfast.new(id: 3, name: "pancakes")
146+
147+
# batch operations
148+
operation = Aws::Record::Batch.write(client: Breakfast.dynamodb_client) do |db|
149+
db.put(waffles)
150+
db.delete(eggs)
151+
db.put(pancakes)
152+
end
153+
154+
# unprocessed items can be retried by calling Aws::Record::BatchWrite#execute!
155+
operation.execute! until operation.complete?
156+
```
157+
92158
### Inheritance Support
93159
Aws Record models can be extended using standard ruby inheritance. The child model must
94160
include `Aws::Record` in their model and the following will be inherited:

features/batch/batch.feature

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
# language: en
2+
3+
@dynamodb @batch
4+
Feature: Amazon DynamoDB Batch
5+
This feature tests the ability to use the batch read and write item APIs via
6+
aws-record. To run these tests, you will need to have valid AWS credentials
7+
that are accessible with the AWS SDK for Ruby's standard credential provider
8+
chain. In practice, this means a shared credential file or environment
9+
variables with your credentials. These tests may have some AWS costs associated
10+
with running them since AWS resources are created and destroyed within
11+
these tests.
12+
13+
Background:
14+
Given a Parent model with definition:
15+
"""
16+
set_table_name('FoodTable')
17+
integer_attr :id, hash_key: true, database_attribute_name: 'Food ID'
18+
string_attr :dish, range_key: true
19+
boolean_attr :spicy
20+
"""
21+
And a Parent model with TableConfig of:
22+
"""
23+
Aws::Record::TableConfig.define do |t|
24+
t.model_class(ParentTableModel)
25+
t.read_capacity_units(2)
26+
t.write_capacity_units(2)
27+
t.client_options(region: "us-east-1")
28+
end
29+
"""
30+
When we migrate the TableConfig
31+
Then eventually the table should exist in DynamoDB
32+
And a Child model with definition:
33+
"""
34+
set_table_name('DessertTable')
35+
boolean_attr :gluten_free
36+
"""
37+
And a Child model with TableConfig of:
38+
"""
39+
Aws::Record::TableConfig.define do |t|
40+
t.model_class(ChildTableModel)
41+
t.read_capacity_units(2)
42+
t.write_capacity_units(2)
43+
t.client_options(region: "us-east-1")
44+
end
45+
"""
46+
When we migrate the TableConfig
47+
Then eventually the table should exist in DynamoDB
48+
49+
Scenario: Perform a batch set of writes and read
50+
When we make a batch write call with following Parent and Child model items:
51+
"""
52+
[
53+
{ "model": "Parent", "id": 1, "dish": "Papaya Salad", "spicy": true },
54+
{ "model": "Parent", "id": 2, "dish": "Hamburger", "spicy": false },
55+
{ "model": "Child", "id": 1, "dish": "Apple Pie", "spicy": false, "gluten_free": false }
56+
]
57+
"""
58+
And we make a batch read call for the following Parent and Child model item keys:
59+
"""
60+
[
61+
{ "model": "Parent", "id": 1, "dish": "Papaya Salad" },
62+
{ "model": "Parent", "id": 2, "dish": "Hamburger" },
63+
{ "model": "Child", "id": 1, "dish": "Apple Pie" }
64+
]
65+
"""
66+
Then we expect the batch read result to include the following items:
67+
"""
68+
[
69+
{ "id": 1, "dish": "Papaya Salad", "spicy": true },
70+
{ "id": 2, "dish": "Hamburger", "spicy": false },
71+
{ "id": 1, "dish": "Apple Pie", "spicy": false, "gluten_free": false }
72+
]
73+
"""

features/batch/step_definitions.rb

+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
# frozen_string_literal: true
2+
3+
And(/^a (Parent|Child) model with TableConfig of:$/) do |model, code_block|
4+
case model
5+
when 'Parent'
6+
ParentTableModel = @parent
7+
when 'Child'
8+
ChildTableModel = @model
9+
else
10+
raise 'Model must be either a Parent or Child'
11+
end
12+
13+
@table_config = eval(code_block)
14+
end
15+
16+
When(/^we make a batch write call with following Parent and Child model items:$/) do |string|
17+
item_data = JSON.parse(string, symbolize_names: true)
18+
19+
Aws::Record::Batch.write do |db|
20+
item_data.each do |item|
21+
case item[:model]
22+
when 'Parent'
23+
db.put(@parent.new(remove_model_key(item)))
24+
when 'Child'
25+
db.put(@model.new(remove_model_key(item)))
26+
else
27+
raise 'Model must be either a Parent or Child'
28+
end
29+
end
30+
end
31+
end
32+
33+
And(/^we make a batch read call for the following Parent and Child model item keys:$/) do |string|
34+
key_batch = JSON.parse(string, symbolize_names: true)
35+
36+
@batch_read_result = Aws::Record::Batch.read do |db|
37+
key_batch.each do |item_key|
38+
case item_key[:model]
39+
when 'Parent'
40+
db.find(@parent, remove_model_key(item_key))
41+
when 'Child'
42+
db.find(@model, remove_model_key(item_key))
43+
else
44+
raise 'Model must be either a Parent or Child'
45+
end
46+
end
47+
end
48+
end
49+
50+
Then(/^we expect the batch read result to include the following items:$/) do |string|
51+
expected = JSON.parse(string, symbolize_names: true)
52+
actual = @batch_read_result.items.map(&:to_h)
53+
expect(expected.count).to eq(actual.count)
54+
expect(expected.all? { |e| actual.include?(e) }).to be_truthy
55+
end
56+
57+
private
58+
59+
def remove_model_key(item)
60+
item.delete(:model)
61+
item
62+
end

lib/aws-record.rb

+2-12
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,4 @@
1-
# Copyright 2015-2017 Amazon.com, Inc. or its affiliates. All Rights Reserved.
2-
#
3-
# Licensed under the Apache License, Version 2.0 (the "License"). You may not
4-
# use this file except in compliance with the License. A copy of the License is
5-
# located at
6-
#
7-
# http://aws.amazon.com/apache2.0/
8-
#
9-
# or in the "license" file accompanying this file. This file is distributed on
10-
# an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
11-
# or implied. See the License for the specific language governing permissions
12-
# and limitations under the License.
1+
# frozen_string_literal: true
132

143
require 'aws-sdk-dynamodb'
154
require_relative 'aws-record/record/client_configuration'
@@ -30,6 +19,7 @@
3019
require_relative 'aws-record/record/version'
3120
require_relative 'aws-record/record/transactions'
3221
require_relative 'aws-record/record/buildable_search'
22+
require_relative 'aws-record/record/batch_read'
3323
require_relative 'aws-record/record/batch_write'
3424
require_relative 'aws-record/record/batch'
3525
require_relative 'aws-record/record/marshalers/string_marshaler'

0 commit comments

Comments
 (0)