How to handle remote services in tests

Do you have difficulties in adding the new tests and their readability decreased due to mocks and stubs? Let’s try to get rid of external requests in tests.

The main idea is to override implementation dynamically during the call of external service. In other words, to use different sources for receiving data for different environments. Suppose for production environments, you get the data from a third-party server, and for a test environment, the source can simply return an object of the desired format.

Bridge Pattern: class with abstraction

First of all, we separate responsibilities in different classes. According to the Bridge pattern, it needs to decouple an abstraction from its implementation.

Here we implement the abstraction with the external service call, and here we will introduce the dependency:

class Medium
  cattr_accessor :source
  
  def initialize(name = nil)
    @client = source.new(name)
  end
  
  def posts
    @client.user_posts
  end
end

cattr_accessor: source allows us to determine which class will participate in the loading of posts.

Bridge Pattern: the Implementators

We need to have implementations: one for the real external service call, another will be called in tests.

Call to real API will look like:

module PostsSource
  class Remote
    def initialize(username)
      @client = MediumAPI.new(username)
    end
    
    def user_posts
      @client.posts
    end
  end
end

And the fake implementation:

module PostsSource
 class Fake
   def initialize(username)
     @client = OpenStruct.new(
       posts: [
         {
           title: 'Signal v Noise exits Medium[Fake source]',
           ...
         }
       ]
     )
   end
   
   def user_posts
     @client.posts
   end
 end
end

Note: All source implementations must have the same interface.

How does it work

When we need to use the external service:

posts = Medium.new('dhh')
posts.source = PostsSource::Remote
posts.user_posts

To use Fake implementation:

posts = Medium.new
posts.source = PostsSource::Fake
posts.user_posts

Settings for various environments

According to Ruby on Rails way, the settings for various environments need to be placed in the initializer.

First, let’s set the default value for the source:

class Medium
  cattr_accessor :source
  self.source = PostsSource::Remote
  
end

For the convenience of testing the code associated with the Medium class, you can make a separate class that will manage the source attribute.

require 'posts_source/fake'

class Medium::Testing
  def self.fake!
    Medium.source = PostsSource::Fake
  end
end

Now in the initializers folder, let’s create the file with necessary configurations:

# frozen_string_literal: true

require 'posts_source/remote'
require 'medium/testing'

Medium::Testing.fake! if Rails.env.test?

In this way PostsSource::Fake will be used for the test environment and PostsSource::Medium the other environment.

Benefits received

The real response still needs to be tested, and most likely, it will be stubbed. But dependency injection allows decreasing the number of stubbing usages in the tests.

Paul Keen is an Open Source Contributor and a Chief Technology Officer at JetThoughts. Follow him on LinkedIn or GitHub.

If you enjoyed this story, we recommend reading our latest tech stories and trending tech stories.