Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Properly group lazy operations in resolve_type #4892

Draft
wants to merge 1 commit into
base: master
Choose a base branch
from
Draft
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
105 changes: 105 additions & 0 deletions spec/graphql/execution/lazy_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -241,4 +241,109 @@ class SubWrapper < LazyHelpers::Wrapper; end
assert_equal(:value, map.get(s))
end
end

describe "Interface.resolve_type" do
class LazyResolveTypeSchema < GraphQL::Schema
class Loader
LOG = []
DATA = {
1 => { versionable: 3 },
2 => { versionable: 4 },
3 => { foo: "foo" },
4 => { bar: "bar" },
}

def initialize(loading_key)
@loading_key = loading_key
@loading_ids = Set.new
@loaded = {}
end

def self.for(context, loading_key)
l_cache = context[:loader_cache] ||= Hash.new { |h, k| h[k] = Loader.new(k) }
l_cache[loading_key]
end

def load(id)
@loading_ids.add(id)
-> {
resolve
result = @loaded.fetch(id)
if block_given?
yield(result)
else
result
end
}
end

def resolve
if @loading_ids.any?
Loader::LOG << [@loading_key, @loading_ids.to_a]
@loading_ids.to_a.each do |id|
@loaded[id] = DATA[id]
end
@loading_ids.clear
end
end
end

module Version
include GraphQL::Schema::Interface

def self.resolve_type(obj, ctx)
Loader.for(ctx, :versionable).load(obj[:versionable]) do |versionable|
versionable[:foo] ? FooVersionable : BarVersionable
end
end
end

class FooVersionable < GraphQL::Schema::Object
implements Version
field :foo, String
end

class BarVersionable < GraphQL::Schema::Object
implements Version
field :bar, String
end

class VersionReference < GraphQL::Schema::Object
field :version, Version

def version
Loader.for(context, :version).load(object[:version])
end
end
class Query < GraphQL::Schema::Object
field :version_references, [VersionReference]

def version_references
[{ version: 1 }, { version: 2 }]
end
end

lazy_resolve(Proc, :call)
query(Query)
orphan_types FooVersionable, BarVersionable
end

it "resolves lazies efficiently" do
LazyResolveTypeSchema::Loader::LOG.clear
query_str = " {
versionReferences {
version {
... on FooVersionable { foo }
... on BarVersionable { bar }
}
}
}"
res = LazyResolveTypeSchema.execute(query_str)
pp res.to_h
expected_log = [

]
assert_equal expected_log, LazyResolveTypeSchema::Loader::LOG
end
end
end
Loading