N1Loader разработан для легкого избежания N+1 проблемы
любого типа. К счатью, гем очень легко интегрировать в GraphQL
API. Без дальнейших отлагательств, давайте рассмотрим простой, но самодостаточный пример.

# Добавляем N1Loader с интеграцией ArLazyPreload 
require "n1_loader/ar_lazy_preload"
require 'graphql'

# Создаем SQLite таблицы в памяти, не относится к библиотеке.
require_relative 'context/setup_database'
# ArLazyPreload требует Rails приложение. Этот код необходим, чтобы избежать этого.
require_relative 'context/setup_ar_lazy'

class User < ActiveRecord::Base
  has_many :payments

  n1_optimized :payments_total do |users|
    # Fetch payments for a group of users we will preload in a single query
    total_per_user = Payment.group(:user_id).where(user: users).sum(:amount).tap { |h| h.default = 0 }

    users.each do |user|
      total = total_per_user[user.id]
      # No promises here, simply add a value for to a user.
      fulfill(user, total)
    end
  end
end

class Payment < ActiveRecord::Base
  belongs_to :user

  validates :amount, presence: true
end

# Запускаем ArLazyPreload глобально
ArLazyPreload.config.auto_preload = true
# Или используем preload_associations_lazily когда загружаем объекты из базы данных

class UserType < GraphQL::Schema::Object
  field :payments_total, Integer
end

class QueryType < GraphQL::Schema::Object
  field  :users, [UserType]

  def users
    User.all
  end
end

class Schema < GraphQL::Schema
  query QueryType
end

query_string = <<~GQL
  {
    users {
      paymentsTotal
    }
  }
GQL

# Исполняем запрос без N+1!
p Schema.execute(query_string)['data']

На этом волшебство N1Loader не заканчивается, так как библиотека так же поддерживает аргументы, которые могут приходить с запросом. И все это без N+1!

class User < ActiveRecord::Base
  n1_optimized :payments_total do
    argument :from
    argument :to

    def perform(users)
      total_per_user =
        Payment
          .group(:user_id)
          .where(created_at: from..to)
          .where(user: users)
          .sum(:amount)
          .tap { |h| h.default = 0 }

      users.each do |user|
        total = total_per_user[user.id]
        fulfill(user, total)
      end
    end
  end
end

class UserType < GraphQL::Schema::Object
  field :payments_total, Integer do
    argument :from, Time
    argument :to, Time
  end
end

query_string = <<~GQL
  {
    users {
      paymentsTotal
    }
  }
GQL

# Нет N+1 и больше не будет!
p Schema.execute(query_string, variables: {from: 3.days.ago, to: 1.day.ago})['data']

Описывайте загрузку данные единожды - и используйте в любой части приложения без N+1!

Посмотрите страницу N1Loader с другими возможностями и попробуйте в своем проекте!

Комментарии (3)


  1. DavidNadejdin
    30.04.2022 23:24
    +1

    Может быть не совсем по теме статьи, но это может быть актуально, даже если проблема n+1 будет решена. Опасность GraphQL заключается скорее в глубине возможной выборки. Для этого глубину ограничивают, так как ресолвер для очень глубокой и сложной выборки, с большей вероятностью положит сервер на лопатки, чем N+1.


    1. aceofspades88
      01.05.2022 11:34
      +1

      все зависит от нагрузки и масштаба N+1, так то на лопатки и он уложить сможет


      1. shushu
        02.05.2022 15:24

        Ну н+1 зависит от размера возвращаемых елементов. И если их ограничить ....