Class: Neo4j::Server::CypherResponse

Inherits:
Object
  • Object
show all
Defined in:
lib/neo4j-server/cypher_response.rb

Defined Under Namespace

Classes: ConstraintViolationError, HashEnumeration, ResponseError

Constant Summary collapse

EMPTY_STRING =
''
RETRYABLE_ERROR_STATUSES =
%w(DeadlockDetectedException AcquireLockTimeoutException ExternalResourceFailureException UnknownFailureException)
CONSTRAINT_ERROR =
'Neo.ClientError.Schema.ConstraintViolation'

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(response, uncommited = false) ⇒ CypherResponse

Returns a new instance of CypherResponse.



136
137
138
139
# File 'lib/neo4j-server/cypher_response.rb', line 136

def initialize(response, uncommited = false)
  @response = response
  @uncommited = uncommited
end

Instance Attribute Details

#columnsObject (readonly)

Returns the value of attribute columns.



4
5
6
# File 'lib/neo4j-server/cypher_response.rb', line 4

def columns
  @columns
end

#dataObject (readonly)

Returns the value of attribute data.



4
5
6
# File 'lib/neo4j-server/cypher_response.rb', line 4

def data
  @data
end

#error_codeObject (readonly)

Returns the value of attribute error_code.



4
5
6
# File 'lib/neo4j-server/cypher_response.rb', line 4

def error_code
  @error_code
end

#error_msgObject (readonly)

Returns the value of attribute error_msg.



4
5
6
# File 'lib/neo4j-server/cypher_response.rb', line 4

def error_msg
  @error_msg
end

#error_statusObject (readonly)

Returns the value of attribute error_status.



4
5
6
# File 'lib/neo4j-server/cypher_response.rb', line 4

def error_status
  @error_status
end

#responseObject (readonly)

Returns the value of attribute response.



4
5
6
# File 'lib/neo4j-server/cypher_response.rb', line 4

def response
  @response
end

#structObject (readonly)

Returns the value of attribute struct.



134
135
136
# File 'lib/neo4j-server/cypher_response.rb', line 134

def struct
  @struct
end

Class Method Details

.create_with_no_tx(response) ⇒ Object



229
230
231
232
233
234
235
236
# File 'lib/neo4j-server/cypher_response.rb', line 229

def self.create_with_no_tx(response)
  case response.status
  when 200 then new(response).set_data(response.body)
  when 400 then new(response).set_error(response.body)
  else
    fail "Unknown response code #{response.status} for #{response.env[:url]}"
  end
end

.create_with_tx(response) ⇒ Object



238
239
240
241
242
243
244
245
# File 'lib/neo4j-server/cypher_response.rb', line 238

def self.create_with_tx(response)
  fail "Unknown response code #{response.status} for #{response.request_uri}" unless response.status == 200

  new(response, true).tap do |cr|
    body = response.body
    body[:errors].empty? ? cr.set_data(body[:results].first) : cr.set_error(body[:errors].first)
  end
end

Instance Method Details

#add_entity_id(data) ⇒ Object



161
162
163
164
165
166
167
168
# File 'lib/neo4j-server/cypher_response.rb', line 161

def add_entity_id(data)
  data[:id] = if data[:metadata] && data[:metadata][:id]
                data[:metadata][:id]
              else
                data[:self].split('/')[-1].to_i
              end
  data
end

#add_transaction_entity_idObject



170
171
172
173
# File 'lib/neo4j-server/cypher_response.rb', line 170

def add_transaction_entity_id
  mapped_rest_data[:id] = mapped_rest_data[:self].split('/').last.to_i
  mapped_rest_data
end

#constraint_error?Boolean

Returns:

  • (Boolean)


219
220
221
# File 'lib/neo4j-server/cypher_response.rb', line 219

def constraint_error?
  @error_code == CONSTRAINT_ERROR || @error_msg.include?('already exists with')
end

#data?Boolean

Returns:

  • (Boolean)


185
186
187
# File 'lib/neo4j-server/cypher_response.rb', line 185

def data?
  !response.body[:data].nil?
end

#each_data_rowObject



193
194
195
# File 'lib/neo4j-server/cypher_response.rb', line 193

def each_data_row
  data.each { |r| yield (@uncommited ? r[:row] : r) }
end

#entity_data(id = nil) ⇒ Object



141
142
143
144
145
146
147
148
149
# File 'lib/neo4j-server/cypher_response.rb', line 141

def entity_data(id = nil)
  if @uncommited
    data = @data.first[:row].first
    data.is_a?(Hash) ? {data: data, id: id} : data
  else
    data = @data[0][0]
    data.is_a?(Hash) ? add_entity_id(data) : data
  end
end

#error?Boolean

Returns:

  • (Boolean)


175
176
177
# File 'lib/neo4j-server/cypher_response.rb', line 175

def error?
  !!@error
end

#first_dataObject



151
152
153
154
155
156
157
158
159
# File 'lib/neo4j-server/cypher_response.rb', line 151

def first_data
  if @uncommited
    @data.first[:row].first
    # data.is_a?(Hash) ? {'data' => data, 'id' => id} : data
  else
    data = @data[0][0]
    data.is_a?(Hash) ? add_entity_id(data) : data
  end
end

#hash_value_as_object(value, session) ⇒ Object



86
87
88
89
90
91
92
93
94
95
96
97
98
99
# File 'lib/neo4j-server/cypher_response.rb', line 86

def hash_value_as_object(value, session)
  if transaction_response?
    add_transaction_entity_id
    data = mapped_rest_data
  elsif [:node, :relationship].include?(identify_entity(value))
    add_entity_id(value)
    data = value
  else
    return value
  end

  basic_obj = (node?(value) ? CypherNode : CypherRelationship).new(session, data)
  unwrapped? ? basic_obj : basic_obj.wrapper
end

#identify_entity(data) ⇒ Object



101
102
103
104
105
106
107
108
109
110
111
112
# File 'lib/neo4j-server/cypher_response.rb', line 101

def identify_entity(data)
  self_string = data[:self]
  if self_string
    if self_string.include?('node')
      :node
    elsif self_string.include?('relationship')
      :relationship
    end
  elsif [:nodes, :relationships, :start, :end, :length].all? { |k| data.key?(k) }
    :path
  end
end

#looks_like_an_object?(value) ⇒ Boolean

Returns:

  • (Boolean)


114
115
116
117
118
119
120
# File 'lib/neo4j-server/cypher_response.rb', line 114

def looks_like_an_object?(value)
  if transaction_response?
    mapped_rest_data[:outgoing_relationships] || (mapped_rest_data[:start] && mapped_rest_data[:properties])
  else
    value[:labels] || value[:type]
  end
end

#map_row_value(value, session) ⇒ Object



76
77
78
79
80
81
82
83
84
# File 'lib/neo4j-server/cypher_response.rb', line 76

def map_row_value(value, session)
  if value.is_a?(Hash) && looks_like_an_object?(value)
    hash_value_as_object(value, session)
  elsif value.is_a?(Array)
    value.map! { |v| map_row_value(v, session) }
  else
    value
  end
end

#node?(value) ⇒ Boolean

Returns:

  • (Boolean)


130
131
132
# File 'lib/neo4j-server/cypher_response.rb', line 130

def node?(value)
  transaction_response? ? !mapped_rest_data[:start] : value[:labels]
end

#raise_cypher_errorObject



223
224
225
226
# File 'lib/neo4j-server/cypher_response.rb', line 223

def raise_cypher_error
  fail 'Tried to raise error without an error' unless @error
  fail Neo4j::Session::CypherError.new(@error_msg, @error_code, @error_status)
end

#raise_errorObject



213
214
215
216
217
# File 'lib/neo4j-server/cypher_response.rb', line 213

def raise_error
  fail 'Tried to raise error without an error' unless @error
  error_class = constraint_error? ? ConstraintViolationError : ResponseError
  fail error_class.new(@error_msg, @error_status, @error_code)
end

#raise_unless_response_code(code) ⇒ Object



189
190
191
# File 'lib/neo4j-server/cypher_response.rb', line 189

def raise_unless_response_code(code)
  fail "Response code #{response.status}, expected #{code} for #{response.headers[:location]}, #{response.body}" unless response.status == code
end

#rest_dataObject



251
252
253
254
# File 'lib/neo4j-server/cypher_response.rb', line 251

def rest_data
  @result_index = @row_index = 0
  mapped_rest_data
end

#rest_data_with_idObject



256
257
258
259
# File 'lib/neo4j-server/cypher_response.rb', line 256

def rest_data_with_id
  rest_data[:id] = mapped_rest_data[:self].split('/').last.to_i
  rest_data
end

#retryable_error?Boolean

Returns:

  • (Boolean)


180
181
182
183
# File 'lib/neo4j-server/cypher_response.rb', line 180

def retryable_error?
  return unless error?
  RETRYABLE_ERROR_STATUSES.include?(@error_status)
end

#row_pair_in_struct(row, session) ⇒ Object



67
68
69
70
71
72
73
74
# File 'lib/neo4j-server/cypher_response.rb', line 67

def row_pair_in_struct(row, session)
  @struct.new.tap do |result|
    row.each_pair do |column, value|
      result[column] = map_row_value(value, session)
      @row_index += 1
    end
  end
end

#set_data(response) ⇒ Object



197
198
199
200
201
202
# File 'lib/neo4j-server/cypher_response.rb', line 197

def set_data(response)
  @data = response[:data]
  @columns = response[:columns]
  @struct = @columns.empty? ? Object.new : Struct.new(*@columns.map(&:to_sym))
  self
end

#set_error(error) ⇒ Object



204
205
206
207
208
209
210
# File 'lib/neo4j-server/cypher_response.rb', line 204

def set_error(error)
  @error = true
  @error_msg = error[:message]
  @error_status = error[:status] || error[:exception] || error[:code]
  @error_code = error[:code] || error[:fullname]
  self
end

#to_node_enumeration(cypher = EMPTY_STRING, session = Neo4j::Session.current) ⇒ Object



56
57
58
59
60
61
62
63
64
65
# File 'lib/neo4j-server/cypher_response.rb', line 56

def to_node_enumeration(cypher = EMPTY_STRING, session = Neo4j::Session.current)
  Enumerator.new do |yielder|
    @result_index = 0
    to_struct_enumeration(cypher).each do |row|
      @row_index = 0
      yielder << row_pair_in_struct(row, session)
      @result_index += 1
    end
  end
end

#to_struct_enumeration(cypher = EMPTY_STRING) ⇒ Object



52
53
54
# File 'lib/neo4j-server/cypher_response.rb', line 52

def to_struct_enumeration(cypher = EMPTY_STRING)
  HashEnumeration.new(self, cypher)
end

#transaction_response?Boolean

Returns:

  • (Boolean)


247
248
249
# File 'lib/neo4j-server/cypher_response.rb', line 247

def transaction_response?
  response.respond_to?(:body) && !response.body[:commit].nil?
end

#unwrapped!Object



122
123
124
# File 'lib/neo4j-server/cypher_response.rb', line 122

def unwrapped!
  @_unwrapped_obj = true
end

#unwrapped?Boolean

Returns:

  • (Boolean)


126
127
128
# File 'lib/neo4j-server/cypher_response.rb', line 126

def unwrapped?
  !!@_unwrapped_obj
end