Skip to content

Commit

Permalink
Auto-paginate embedded resources, closes #106.
Browse files Browse the repository at this point in the history
  • Loading branch information
dblock committed Dec 2, 2020
1 parent ed9a796 commit c6fb9e5
Show file tree
Hide file tree
Showing 8 changed files with 152 additions and 14 deletions.
14 changes: 7 additions & 7 deletions .rubocop_todo.yml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# This configuration was generated by
# `rubocop --auto-gen-config`
# on 2020-05-14 17:08:39 -0400 using RuboCop version 0.81.0.
# on 2020-12-02 13:56:49 -0500 using RuboCop version 0.81.0.
# The point is for the user to remove these configuration records
# one by one as the offenses are removed from the code base.
# Note that changes in the inspected code, or installation of new
Expand All @@ -9,14 +9,14 @@
# Offense count: 1
# Configuration parameters: CountComments.
Metrics/ClassLength:
Max: 103
Max: 112

# Offense count: 4
# Offense count: 9
# Configuration parameters: CountComments, ExcludedMethods.
Metrics/MethodLength:
Max: 25
Max: 26

# Offense count: 3
# Offense count: 4
# Configuration parameters: CountComments.
Metrics/ModuleLength:
Max: 265
Expand Down Expand Up @@ -59,9 +59,9 @@ Style/MethodMissingSuper:
Exclude:
- 'lib/hyperclient/collection.rb'

# Offense count: 94
# Offense count: 101
# Cop supports --auto-correct.
# Configuration parameters: AutoCorrect, AllowHeredoc, AllowURI, URISchemes, IgnoreCopDirectives, IgnoredPatterns.
# URISchemes: http, https
Layout/LineLength:
Max: 142
Max: 147
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ api.connection.use :http_cache

## Resources and Attributes

Hyperclient will fetch and discover the resources from your API.
Hyperclient will fetch and discover the resources from your API and automatically paginate when possible.

```ruby
api.splines.each do |spline|
Expand Down
5 changes: 5 additions & 0 deletions features/api_navigation.feature
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,11 @@ Feature: API navigation
When I connect to the API
Then I should be able to navigate to posts and authors

Scenario: Links
When I connect to the API
Then I should be able to paginate posts
Then I should be able to paginate authors

Scenario: Templated links
Given I connect to the API
When I search for a post with a templated link
Expand Down
20 changes: 19 additions & 1 deletion features/steps/api_navigation.rb
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,24 @@ class Spinach::Features::ApiNavigation < Spinach::FeatureSteps
assert_requested :get, 'http://api.example.org/authors'
end

step 'I should be able to paginate posts' do
count = 0
api.posts.each do |_post|
count += 1
end
assert_requested :get, 'http://api.example.org/posts'
assert_requested :get, 'http://api.example.org/posts?page=2'
assert_requested :get, 'http://api.example.org/posts?page=3'
end

step 'I should be able to paginate authors' do
api._links['api:authors'].each do |author|
p author
end

assert_requested :get, 'http://api.example.org/authors'
end

step 'I search for a post with a templated link' do
api._links.search._expand(q: 'something')._resource
end
Expand Down Expand Up @@ -59,6 +77,6 @@ class Spinach::Features::ApiNavigation < Spinach::FeatureSteps
api.posts.each do |_post|
count += 1
end
assert_equal 2, count
assert_equal 4, count
end
end
4 changes: 2 additions & 2 deletions features/steps/default_config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ class Spinach::Features::DefaultConfig < Spinach::FeatureSteps
end

step 'it should have been parsed as JSON' do
@posts._attributes.total_posts.to_i.must_equal 2
@posts._attributes['total_posts'].to_i.must_equal 2
@posts._attributes.total_posts.to_i.must_equal 4
@posts._attributes['total_posts'].to_i.must_equal 4
end
end
8 changes: 7 additions & 1 deletion features/support/api.rb
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,14 @@ module API
WebMock::Config.instance.query_values_notation = :flat_array

stub_request(:any, /api.example.org*/).to_return(body: root_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org').to_return(body: root_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/authors').to_return(body: authors_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/posts').to_return(body: posts_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/posts/1').to_return(body: post_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/posts?page=2').to_return(body: posts_page2_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/posts?page=3').to_return(body: posts_page3_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/posts/1').to_return(body: post1_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/posts/2').to_return(body: post2_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/posts/3').to_return(body: post3_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/page2').to_return(body: page2_response, headers: { 'Content-Type' => 'application/hal+json' })
stub_request(:get, 'api.example.org/page3').to_return(body: page3_response, headers: { 'Content-Type' => 'application/hal+json' })
end
Expand Down
98 changes: 96 additions & 2 deletions features/support/fixtures.rb
Original file line number Diff line number Diff line change
Expand Up @@ -15,13 +15,32 @@ def root_response
}'
end

def authors_response
'{
"_links": {
"self": { "href": "/authors" }
},
"_embedded": {
"authors": [
{
"name": "Lorem Ipsum",
"_links": {
"self": { "href": "/authors/1" }
}
}
]
}
}'
end

def posts_response
'{
"_links": {
"self": { "href": "/posts" },
"next": {"href": "/posts?page=2"},
"last_post": {"href": "/posts/1"}
},
"total_posts": "2",
"total_posts": "4",
"_embedded": {
"posts": [
{
Expand All @@ -43,7 +62,48 @@ def posts_response
}'
end

def post_response
def posts_page2_response
'{
"_links": {
"self": { "href": "/posts?page=2" },
"next": { "href": "/posts?page=3" }
},
"total_posts": "4",
"_embedded": {
"posts": [
{
"title": "My third blog post",
"body": "Lorem ipsum dolor sit amet",
"_links": {
"self": { "href": "/posts/3" }
}
}
]
}
}'
end

def posts_page3_response
'{
"_links": {
"self": { "href": "/posts?page=3" }
},
"total_posts": "4",
"_embedded": {
"posts": [
{
"title": "My third blog post",
"body": "Lorem ipsum dolor sit amet",
"_links": {
"self": { "href": "/posts/4" }
}
}
]
}
}'
end

def post1_response
'{
"_links": {
"self": { "href": "/posts/1" }
Expand All @@ -60,6 +120,40 @@ def post_response
}'
end

def post2_response
'{
"_links": {
"self": { "href": "/posts/2" }
},
"title": "My first blog post",
"body": "Lorem ipsum dolor sit amet",
"_embedded": {
"comments": [
{
"title": "Some comment"
}
]
}
}'
end

def post3_response
'{
"_links": {
"self": { "href": "/posts/3" }
},
"title": "My first blog post",
"body": "Lorem ipsum dolor sit amet",
"_embedded": {
"comments": [
{
"title": "Some comment"
}
]
}
}'
end

def page2_response
'{
"_links": {
Expand Down
15 changes: 15 additions & 0 deletions lib/hyperclient/link.rb
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,21 @@ def initialize(key, link, entry_point, uri_variables = nil)
@resource = nil
end

# Public: Each implementation to allow the class to use the Enumerable
# benefits for paginated, embedded items.
#
# Returns an Enumerator.
def each(&block)
current = self
while current
coll = current.respond_to?(@key) ? current.send(@key) : _resource
coll.each(&block)
break unless current._links[:next]

current = current._links.next
end
end

# Public: Indicates if the link is an URITemplate or a regular URI.
#
# Returns true if it is templated.
Expand Down

0 comments on commit c6fb9e5

Please sign in to comment.