Class: Sequel::Postgres::PGArray::Parser
- Defined in:
- lib/sequel/extensions/pg_array.rb
Overview
PostgreSQL array parser that handles PostgreSQL array output format. Note that does not handle all forms out input that PostgreSQL will accept, and it will not raise an error for all forms of invalid input.
Instance Method Summary collapse
-
#initialize(source, converter = nil) ⇒ Parser
constructor
Set the source for the input, and any converter callable to call with objects to be created.
-
#new_entry(include_empty = false) ⇒ Object
Take the buffer of recorded characters and add it to the array of entries, and use a new buffer for recorded characters.
-
#parse ⇒ Object
Parse the input character by character, returning an array of parsed (and potentially converted) objects.
Constructor Details
#initialize(source, converter = nil) ⇒ Parser
Set the source for the input, and any converter callable to call with objects to be created. For nested parsers the source may contain text after the end current parse, which will be ignored.
336 337 338 339 340 341 342 |
# File 'lib/sequel/extensions/pg_array.rb', line 336 def initialize(source, converter=nil) super(source) @converter = converter @stack = [[]] @encoding = string.encoding @recorded = String.new.force_encoding(@encoding) end |
Instance Method Details
#new_entry(include_empty = false) ⇒ Object
Take the buffer of recorded characters and add it to the array of entries, and use a new buffer for recorded characters.
346 347 348 349 350 351 352 353 354 355 356 357 |
# File 'lib/sequel/extensions/pg_array.rb', line 346 def new_entry(include_empty=false) if !@recorded.empty? || include_empty entry = @recorded if entry == 'NULL' && !include_empty entry = nil elsif @converter entry = @converter.call(entry) end @stack.last.push(entry) @recorded = String.new.force_encoding(@encoding) end end |
#parse ⇒ Object
Parse the input character by character, returning an array of parsed (and potentially converted) objects.
361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 |
# File 'lib/sequel/extensions/pg_array.rb', line 361 def parse raise Sequel::Error, "invalid array, empty string" if eos? raise Sequel::Error, "invalid array, doesn't start with {" unless scan(/((\[\d+:\d+\])+=)?\{/) # :nocov: while !eos? # :nocov: char = scan(/[{}",]|[^{}",]+/) if char == ',' # Comma outside quoted string indicates end of current entry new_entry elsif char == '"' raise Sequel::Error, "invalid array, opening quote with existing recorded data" unless @recorded.empty? # :nocov: while true # :nocov: char = scan(/["\\]|[^"\\]+/) if char == '\\' @recorded << getch elsif char == '"' n = peek(1) raise Sequel::Error, "invalid array, closing quote not followed by comma or closing brace" unless n == ',' || n == '}' break else @recorded << char end end new_entry(true) elsif char == '{' raise Sequel::Error, "invalid array, opening brace with existing recorded data" unless @recorded.empty? # Start of new array, add it to the stack new = [] @stack.last << new @stack << new elsif char == '}' # End of current array, add current entry to the current array new_entry if @stack.length == 1 raise Sequel::Error, "array parsing finished without parsing entire string" unless eos? # Top level of array, parsing should be over. # Pop current array off stack and return it as result return @stack.pop else # Nested array, pop current array off stack @stack.pop end else # Add the character to the recorded character buffer. @recorded << char end end raise Sequel::Error, "array parsing finished with array unclosed" end |