Commit 2a100641 authored by charlieablett's avatar charlieablett

Restructure complexity analyzer

Remove instance variables for class re-use, test individual methods,
use `monotonic_time`
parent 2c011cb5
...@@ -4,10 +4,6 @@ module Gitlab ...@@ -4,10 +4,6 @@ module Gitlab
module Graphql module Graphql
module QueryAnalyzers module QueryAnalyzers
class LoggerAnalyzer class LoggerAnalyzer
def initialize
@info_hash = {}
end
# Called before initializing the analyzer. # Called before initializing the analyzer.
# Returns true to run this analyzer, or false to skip it. # Returns true to run this analyzer, or false to skip it.
def analyze?(query) def analyze?(query)
...@@ -18,16 +14,20 @@ module Gitlab ...@@ -18,16 +14,20 @@ module Gitlab
# Returns the initial value for `memo` # Returns the initial value for `memo`
def initial_value(query) def initial_value(query)
{ {
time_started: Time.zone.now, time_started: Gitlab::Metrics::System.monotonic_time,
query_string: query.query_string, query_string: query.query_string,
variables: query.provided_variables variables: process_variables(query.provided_variables),
complexity: nil,
depth: nil,
duration: nil
} }
end end
# This is like the `reduce` callback. # This is like the `reduce` callback.
# The return value is passed to the next call as `memo` # The return value is passed to the next call as `memo`
def call(memo, visit_type, irep_node) def call(memo, visit_type, irep_node)
memo memo = set_complexity(memo)
set_depth(memo)
end end
# Called when we're done the whole visit. # Called when we're done the whole visit.
...@@ -35,29 +35,37 @@ module Gitlab ...@@ -35,29 +35,37 @@ module Gitlab
# Or, you can use this hook to write to a log, etc # Or, you can use this hook to write to a log, etc
def final_value(memo) def final_value(memo)
memo[:duration] = "#{duration(memo[:time_started]).round(1)}ms" memo[:duration] = "#{duration(memo[:time_started]).round(1)}ms"
set_complexity GraphqlLogger.info(memo.except!(:time_started))
set_depth
GraphqlLogger.info(memo.except!(:time_started).merge(@info_hash))
memo memo
end end
private private
def set_complexity def process_variables(variables)
if variables.respond_to?(:to_unsafe_h)
variables.to_unsafe_h
else
variables
end
end
def set_complexity(memo)
GraphQL::Analysis::QueryComplexity.new do |query, complexity_value| GraphQL::Analysis::QueryComplexity.new do |query, complexity_value|
@info_hash[:complexity] = complexity_value memo[:complexity] = complexity_value
end end
memo
end end
def set_depth def set_depth(memo)
GraphQL::Analysis::QueryDepth.new do |query, depth_value| GraphQL::Analysis::QueryDepth.new do |query, depth_value|
@info_hash[:depth] = depth_value memo[:depth] = depth_value
end end
memo
end end
def duration(time_started) def duration(time_started)
nanoseconds = Time.zone.now - time_started nanoseconds = Gitlab::Metrics::System.monotonic_time - time_started
nanoseconds / 1000000 nanoseconds * 1000000
end end
end end
end end
......
...@@ -7,21 +7,43 @@ describe Gitlab::Graphql::QueryAnalyzers::LoggerAnalyzer do ...@@ -7,21 +7,43 @@ describe Gitlab::Graphql::QueryAnalyzers::LoggerAnalyzer do
subject { described_class.new } subject { described_class.new }
let(:query_string) { "abc" } let(:query_string) { "abc" }
let(:provided_variables) { { a: 1, b: 2, c: 3 } } let(:provided_variables) { { a: 1, b: 2, c: 3 } }
let!(:now) { Gitlab::Metrics::System.monotonic_time }
let(:complexity) { 4 } let(:complexity) { 4 }
let(:depth) { 2 } let(:depth) { 2 }
let(:expected_hash) do let(:initial_values) do
{ query_string: query_string, { time_started: now,
query_string: query_string,
variables: provided_variables, variables: provided_variables,
complexity: complexity, complexity: nil,
depth: depth } depth: nil,
duration: nil }
end
before do
allow(Gitlab::Metrics::System).to receive(:monotonic_time).and_return(now)
end end
it 'assembles a hash' do describe '#initial_value' do
it 'assembles a hash with initial values' do
query = OpenStruct.new(query_string: query_string, provided_variables: provided_variables) query = OpenStruct.new(query_string: query_string, provided_variables: provided_variables)
subject.initial_value(query) expect(subject.initial_value(query)).to eq initial_values
end
end
expect(subject.instance_variable_get("@info_hash")).to eq expected_hash describe '#call' do
before do
# some statements to fudge the complexity and depth
end end
it 'sets the complexity and depth' do
expected_hash = { time_started: now,
query_string: query_string,
variables: provided_variables,
complexity: nil,
depth: depth,
duration: complexity }
expect(subject.call(initial_values, nil, nil)).to eq expected_hash
end
end
end end
...@@ -85,7 +85,7 @@ describe 'GitlabSchema configurations' do ...@@ -85,7 +85,7 @@ describe 'GitlabSchema configurations' do
context 'logging' do context 'logging' do
it 'writes to the GraphQL log' do it 'writes to the GraphQL log' do
expect(Gitlab::GraphqlLogger).to receive(:info).with(/Query Complexity/) expect(Gitlab::GraphqlLogger).to receive(:info)
query = File.read(Rails.root.join('spec/fixtures/api/graphql/introspection.graphql')) query = File.read(Rails.root.join('spec/fixtures/api/graphql/introspection.graphql'))
......
...@@ -17,14 +17,32 @@ describe 'GraphQL' do ...@@ -17,14 +17,32 @@ describe 'GraphQL' do
end end
context 'logging' do context 'logging' do
it 'logs the query' do before do
expected = { query_string: query, variables: {}, duration: anything }
expect(Gitlab::GraphqlLogger).to receive(:info).with(expected) expect(Gitlab::GraphqlLogger).to receive(:info).with(expected)
end
context 'with no variables' do
let(:expected) do
{ query_string: query, variables: {}, duration: anything, depth: 0, complexity: 0 }
end
it 'logs the query' do
post_graphql(query) post_graphql(query)
end end
end
context 'with variables' do
let!(:variables) do
{ foo: "bar" }
end
let(:expected) do
{ query_string: query, variables: variables, duration: anything, depth: 0, complexity: 0 }
end
it 'logs the query' do
post_graphql(query, variables: variables)
end
end
end end
context 'invalid variables' do context 'invalid variables' do
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment