Sha256: 39fbfda843bbe54d6638527a78360cf55f430dca93d886a87af6bb9fef317b87

Contents?: true

Size: 1.65 KB

Versions: 2

Compression:

Stored size: 1.65 KB

Contents

# A sample HTTP loader using:
#
# 1. https://github.com/httprb/http
# 2. https://github.com/mperham/connection_pool
# 3. https://github.com/ruby-concurrency/concurrent-ruby
#
# Setup:
#
#   field :weather, String, null: true do
#     argument :lat, Float, required: true
#     argument :lng, Float, required: true
#     argument :lang, String, required: true
#   end
#
#   def weather(lat:, lng:, lang:)
#     key = Rails.application.credentials.darksky_key
#     path = "/forecast/#{key}/#{lat},#{lng}?lang=#{lang}"
#     Loaders::HTTPLoader
#       .for(host: 'https://api.darksky.net')
#       .load(->(connection) { connection.get(path).flush })
#       .then do |response|
#         if response.status.ok?
#           json = JSON.parse(response.body)
#           json['currently']['summary']
#         end
#       end
#   end
#
# Querying:
#
#   <<~GQL
#     query Weather {
#       montreal: weather(lat: 45.5017, lng: -73.5673, lang: "fr")
#       waterloo: weather(lat: 43.4643, lng: -80.5204, lang: "en")
#     }
#   GQL

module Loaders
  class HTTPLoader < GraphQL::Batch::Loader
    def initialize(host:, size: 4, timeout: 4)
      @host = host
      @size = size
      @timeout = timeout
    end

    def perform(operations)
      futures = operations.map do |operation|
        Concurrent::Promises.future do
          pool.with { |connection| operation.call(connection) }
        end
      end
      operations.each_with_index.each do |operation, index|
        fulfill(operation, futures[index].value)
      end
    end

  private

    def pool
      @pool ||= ConnectionPool.new(size: @size, timeout: @timeout) do
        HTTP.persistent(@host)
      end
    end
  end
end

Version data entries

2 entries across 2 versions & 1 rubygems

Version Path
graphql-batch-0.4.3 examples/http_loader.rb
graphql-batch-0.4.2 examples/http_loader.rb