mirror of
https://github.com/zammad/zammad
synced 2026-05-24 09:48:36 +00:00
254 lines
7.1 KiB
Ruby
254 lines
7.1 KiB
Ruby
# Copyright (C) 2012-2026 Zammad Foundation, https://zammad-foundation.org/
|
|
|
|
require 'graphql/gql/shared_examples/fails_if_unauthenticated'
|
|
|
|
module ZammadSpecSupportGraphql
|
|
#
|
|
# A stub implementation of ActionCable.
|
|
# Any methods to support the mock backend have `mock` in the name.
|
|
# Taken from github.com/rmosolgo/graphql-ruby/blob/master/spec/graphql/subscriptions/action_cable_subscriptions_spec.rb
|
|
#
|
|
class MockActionCable
|
|
class MockChannel
|
|
def initialize
|
|
@mock_broadcasted_messages = []
|
|
end
|
|
|
|
attr_reader :mock_broadcasted_messages
|
|
|
|
def stream_from(stream_name, coder: nil, &block)
|
|
# Rails uses `coder`, we don't
|
|
block ||= ->(msg) { @mock_broadcasted_messages << msg }
|
|
MockActionCable.mock_stream_for(stream_name).add_mock_channel(self, block)
|
|
end
|
|
|
|
def mock_broadcasted_at(index)
|
|
data = mock_broadcasted_messages.dig(index, :result)
|
|
|
|
return if !data
|
|
|
|
GraphQLHelpers::Result.new(data)
|
|
end
|
|
|
|
def mock_broadcasted_first
|
|
mock_broadcasted_at(0)
|
|
end
|
|
end
|
|
|
|
class MockStream
|
|
def initialize
|
|
@mock_channels = {}
|
|
end
|
|
|
|
def add_mock_channel(channel, handler)
|
|
@mock_channels[channel] = handler
|
|
end
|
|
|
|
def mock_broadcast(message)
|
|
@mock_channels.each_value do |handler|
|
|
handler&.call(message)
|
|
end
|
|
end
|
|
end
|
|
|
|
class << self
|
|
def clear_mocks
|
|
@mock_streams = {}
|
|
end
|
|
|
|
def server
|
|
self
|
|
end
|
|
|
|
def broadcast(stream_name, message)
|
|
stream = @mock_streams[stream_name]
|
|
stream&.mock_broadcast(message)
|
|
end
|
|
|
|
def mock_stream_for(stream_name)
|
|
@mock_streams[stream_name] ||= MockStream.new
|
|
end
|
|
|
|
def build_mock_channel
|
|
MockChannel.new
|
|
end
|
|
|
|
def mock_stream_names
|
|
@mock_streams.keys
|
|
end
|
|
end
|
|
end
|
|
|
|
#
|
|
# Create a mock channel that can be passed to graphql_execute like this:
|
|
#
|
|
# let(:mock_channel) { build_mock_channel }
|
|
#
|
|
# gql.execute(query, context: { channel: mock_channel })
|
|
#
|
|
delegate :build_mock_channel, to: MockActionCable
|
|
|
|
#
|
|
# A set of GraphQL helpers. Access them in a :graphql test via `gql.*`.
|
|
#
|
|
class GraphQLHelpers
|
|
|
|
#
|
|
# Encapsulates the GraphQL result.
|
|
#
|
|
# gql.result.*
|
|
#
|
|
class Result
|
|
attr_reader :payload
|
|
|
|
def initialize(payload)
|
|
@payload = payload.with_indifferent_access
|
|
end
|
|
|
|
#
|
|
# Access the data payload. This asserts that only one operation was executed
|
|
# and that no errors are present.
|
|
#
|
|
# expect(gql.response.data).to include(...)
|
|
#
|
|
def data
|
|
assert('GraphQL result does not contain errors') do
|
|
@payload[:errors].nil?
|
|
end
|
|
assert('GraphQL result contains exactly one data entry') do
|
|
@payload[:data]&.one?
|
|
end
|
|
@payload[:data].values.first
|
|
end
|
|
|
|
#
|
|
# Access the edges->node data payload from `#data()` in a convenient way.
|
|
#
|
|
# expect(gql.response.nodes.first).to include(...)
|
|
#
|
|
# Also can operate on a subentry in the hash rather than the top level.
|
|
#
|
|
# expect(gql.response.nodes('first_level', 'second_level').first).to include(...)
|
|
#
|
|
def nodes(*subkeys)
|
|
content = data.dig(*subkeys, :edges)
|
|
assert('GraphQL result contains node entries') do
|
|
!content.nil?
|
|
end
|
|
content.pluck(:node)
|
|
end
|
|
|
|
#
|
|
# Access an error entry. This asserts that only one error and no data payload is present.
|
|
#
|
|
# expect(gql.result.error).to include(...)
|
|
#
|
|
def error
|
|
assert('GraphQL result does not contain data') do
|
|
@payload[:data].nil? || @payload[:data].values.first.nil?
|
|
end
|
|
assert('GraphQL result contains exactly one error entry') do
|
|
@payload[:errors]&.one?
|
|
end
|
|
@payload[:errors][0]
|
|
end
|
|
|
|
#
|
|
# Access the error type from `#error()` and return it as a Ruby class.
|
|
#
|
|
# expect(gql.result.error_type).to eq(ActiveRecord::RecordNotFound)
|
|
#
|
|
def error_type
|
|
assert('GraphQL result has error type') do
|
|
error.dig(:extensions, :type).present?
|
|
end
|
|
error.dig(:extensions, :type).constantize
|
|
end
|
|
|
|
#
|
|
# Access the error message from `#error()`.
|
|
#
|
|
# expect(gql.result.error_message).to eq('Something went wrong in this test...')
|
|
#
|
|
def error_message
|
|
error[:message]
|
|
end
|
|
|
|
private
|
|
|
|
def assert(message)
|
|
raise "Assertion '#{message}' failed, graphql result:\n#{PP.pp(payload, '')}" if !yield
|
|
end
|
|
end
|
|
|
|
attr_writer :graphql_current_user
|
|
attr_accessor :result
|
|
|
|
# Shortcut to generate a GraphQL ID for an object.
|
|
def id(object)
|
|
Gql::ZammadSchema.id_from_object(object)
|
|
end
|
|
|
|
#
|
|
# Run a graphql query.
|
|
#
|
|
# before do
|
|
# gql.execute(query, variables: { ... })
|
|
# end
|
|
#
|
|
# Afterwards, the `Result` can be accessed via
|
|
#
|
|
# gql.result
|
|
#
|
|
def execute(query, variables: {}, context: {})
|
|
context[:controller] ||= GraphqlController.new
|
|
.tap do |controller|
|
|
controller.request = ActionDispatch::Request.new({})
|
|
controller.request.remote_ip = context[:REMOTE_IP] || '127.0.0.1'
|
|
end
|
|
|
|
context[:current_user] ||= @graphql_current_user
|
|
context[:current_user_id] ||= @graphql_current_user&.id
|
|
if @graphql_current_user
|
|
# TODO: we only fake a SID for now, create a real session?
|
|
context[:sid] = SecureRandom.hex(16)
|
|
|
|
# we need to set the current_user_id in the UserInfo context as well
|
|
UserInfo.current_user_id = context[:current_user].id
|
|
end
|
|
@result = Result.new(Gql::ZammadSchema.execute(query, variables: variables, context: context).to_h)
|
|
end
|
|
end
|
|
|
|
def gql
|
|
@gql ||= GraphQLHelpers.new
|
|
end
|
|
end
|
|
|
|
RSpec.configure do |config|
|
|
config.include ZammadSpecSupportGraphql, type: :graphql
|
|
|
|
config.prepend_before(:each, type: :graphql) do
|
|
ZammadSpecSupportGraphql::MockActionCable.clear_mocks
|
|
Gql::ZammadSchema.subscriptions = GraphQL::Subscriptions::ActionCableSubscriptions.new(
|
|
action_cable: ZammadSpecSupportGraphql::MockActionCable, action_cable_coder: JSON, schema: Gql::ZammadSchema
|
|
)
|
|
end
|
|
|
|
config.append_after(:each, type: :graphql) do
|
|
Gql::ZammadSchema.subscriptions = GraphQL::Subscriptions::ActionCableSubscriptions.new(schema: Gql::ZammadSchema)
|
|
end
|
|
|
|
# This helper allows you to authenticate as a given user in :graphql specs
|
|
# via the example metadata, rather than directly:
|
|
#
|
|
# it 'does something', authenticated_as: :user
|
|
#
|
|
# In order for this to work, you must define the user in a `let` block first:
|
|
#
|
|
# let(:user) { create(:customer) }
|
|
#
|
|
config.before(:each, :authenticated_as, type: :graphql) do |example|
|
|
gql.graphql_current_user = authenticated_as_get_user example.metadata[:authenticated_as], return_type: :user
|
|
end
|
|
end
|