marc 1.1.1 → 1.3.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (52) hide show
  1. checksums.yaml +4 -4
  2. data/.github/ISSUE_TEMPLATE/bug_report.md +30 -0
  3. data/.github/workflows/ruby.yml +24 -0
  4. data/.gitignore +17 -0
  5. data/.standard.yml +1 -0
  6. data/{Changes → CHANGELOG.md} +116 -30
  7. data/Gemfile +5 -0
  8. data/README.md +239 -46
  9. data/Rakefile +14 -14
  10. data/bin/marc +14 -0
  11. data/bin/marc2xml +17 -0
  12. data/examples/xml2marc.rb +10 -0
  13. data/lib/marc/constants.rb +3 -3
  14. data/lib/marc/controlfield.rb +35 -23
  15. data/lib/marc/datafield.rb +70 -63
  16. data/lib/marc/dublincore.rb +59 -41
  17. data/lib/marc/exception.rb +9 -1
  18. data/lib/marc/jsonl_reader.rb +33 -0
  19. data/lib/marc/jsonl_writer.rb +44 -0
  20. data/lib/marc/marc8/map_to_unicode.rb +16417 -16420
  21. data/lib/marc/marc8/to_unicode.rb +80 -87
  22. data/lib/marc/reader.rb +116 -124
  23. data/lib/marc/record.rb +72 -62
  24. data/lib/marc/subfield.rb +12 -10
  25. data/lib/marc/unsafe_xmlwriter.rb +93 -0
  26. data/lib/marc/version.rb +1 -1
  27. data/lib/marc/writer.rb +27 -30
  28. data/lib/marc/xml_parsers.rb +222 -197
  29. data/lib/marc/xmlreader.rb +131 -114
  30. data/lib/marc/xmlwriter.rb +93 -82
  31. data/lib/marc.rb +20 -18
  32. data/marc.gemspec +28 -0
  33. data/test/marc8/tc_marc8_mapping.rb +3 -3
  34. data/test/marc8/tc_to_unicode.rb +28 -34
  35. data/test/messed_up_leader.xml +9 -0
  36. data/test/tc_controlfield.rb +37 -34
  37. data/test/tc_datafield.rb +65 -60
  38. data/test/tc_dublincore.rb +9 -11
  39. data/test/tc_hash.rb +10 -13
  40. data/test/tc_jsonl.rb +19 -0
  41. data/test/tc_marchash.rb +17 -21
  42. data/test/tc_parsers.rb +108 -144
  43. data/test/tc_reader.rb +35 -36
  44. data/test/tc_reader_char_encodings.rb +149 -169
  45. data/test/tc_record.rb +143 -148
  46. data/test/tc_subfield.rb +14 -13
  47. data/test/tc_unsafe_xml.rb +95 -0
  48. data/test/tc_writer.rb +101 -108
  49. data/test/tc_xml.rb +101 -94
  50. data/test/tc_xml_error_handling.rb +7 -8
  51. data/test/ts_marc.rb +8 -8
  52. metadata +129 -22
@@ -1,152 +1,187 @@
1
1
  module MARC
2
2
  # Exception class to be thrown when an XML parser
3
3
  # encounters an unrecoverable error.
4
- class XMLParseError < StandardError; end
4
+ class XMLParseError < StandardError
5
+ end
6
+
7
+ IND1 = "ind1".freeze
8
+ IND2 = "ind2".freeze
9
+ TAG = "tag".freeze
10
+ CODE = "code".freeze
5
11
 
6
12
  # The MagicReader will try to use the best available XML Parser at the
7
- # time of initialization.
13
+ # time of initialization.
8
14
  # The order is currently:
9
15
  # * Nokogiri
10
- # * jrexml (JRuby only)
16
+ # * libxml-ruby (MRI only) ** DEPRECATED **
17
+ # * jstax (JRuby only) ** DEPRECATED **
11
18
  # * rexml
12
19
  #
13
20
  # With the idea that other parsers could be added as their modules are
14
21
  # added. Realistically, this list should be limited to stream-based
15
22
  # parsers. The magic should be used selectively, however. After all,
16
23
  # one project's definition of 'best' might not apply universally. It
17
- # is arguable which is "best" on JRuby: Nokogiri or jrexml.
18
- module MagicReader
24
+ # is arguable which is "best" on JRuby: Nokogiri or jrexml.
25
+ module MagicReader
19
26
  def self.extended(receiver)
20
27
  magic = MARC::XMLReader.best_available
21
28
  case magic
22
- when 'nokogiri' then receiver.extend(NokogiriReader)
23
- when 'libxml' then receiver.extend(LibXMLReader)
24
- when 'jstax' then receiver.extend(JRubySTAXReader)
25
- when 'jrexml' then receiver.extend(JREXMLReader)
29
+ when "nokogiri"
30
+ receiver.extend(NokogiriReader)
31
+ when "libxml"
32
+ warn "libxml support will be removed in version 1.3. Prefer nokogiri instead"
33
+ receiver.extend(LibXMLReader)
34
+ when "jstax"
35
+ warn "jstax support will be removed in version 1.3. Prefer nokogiri instead"
36
+ receiver.extend(JRubySTAXReader)
37
+ when "jrexml"
38
+ warn "jrexml support is broken upstream; falling back to just rexml. Prefer nokogiri instead"
39
+ receiver.extend(REXMLReader)
26
40
  else receiver.extend(REXMLReader)
27
41
  end
28
42
  end
29
43
  end
30
-
44
+
31
45
  module GenericPullParser
32
46
  # Submodules must include
33
47
  # self.extended()
34
48
  # init()
35
49
  # attributes_to_hash(attributes)
36
50
  # each
37
-
51
+
52
+ REC_TAG = "record".freeze
53
+ LEAD_TAG = "leader".freeze
54
+ CF_TAG = "controlfield".freeze
55
+ DF_TAG = "datafield".freeze
56
+ SF_TAG = "subfield".freeze
57
+
58
+ def init
59
+ @record = {record: nil, leader: "", field: nil, subfield: nil}
60
+ @current_element = nil
61
+ @ns = "http://www.loc.gov/MARC21/slim"
62
+ end
38
63
 
39
64
  # Returns our MARC::Record object to the #each block.
40
65
  def yield_record
41
- @block.call(@record[:record])
66
+ if @record[:record].valid?
67
+ @block.call(@record[:record])
68
+ elsif @error_handler
69
+ @error_handler.call(self, @record[:record], @block)
70
+ else
71
+ raise MARC::RecordException, @record[:record]
72
+ end
73
+ ensure
42
74
  @record[:record] = nil
43
- end
75
+ end
44
76
 
45
77
  def start_element_namespace name, attributes = [], prefix = nil, uri = nil, ns = {}
46
- attributes = attributes_to_hash(attributes)
47
- if uri == @ns
48
- case name.downcase
49
- when 'record' then @record[:record] = MARC::Record.new
50
- when 'leader' then @current_element = :leader
51
- when 'controlfield'
52
- @current_element=:field
53
- @record[:field] = MARC::ControlField.new(attributes["tag"])
54
- when 'datafield'
55
- @record[:field] = MARC::DataField.new(attributes["tag"], attributes['ind1'], attributes['ind2'])
56
- when 'subfield'
57
- @current_element=:subfield
58
- @record[:subfield] = MARC::Subfield.new(attributes['code'])
59
- end
60
- end
61
- end
62
-
63
-
64
- def characters text
78
+ attributes = attributes_to_hash(attributes)
79
+ if (uri == @ns) || @ignore_namespace
80
+ case name.downcase
81
+ when SF_TAG
82
+ @current_element = :subfield
83
+ @record[:subfield] = MARC::Subfield.new(attributes[CODE])
84
+ when DF_TAG
85
+ @record[:field] = MARC::DataField.new(attributes[TAG], attributes[IND1], attributes[IND2])
86
+ when CF_TAG
87
+ @current_element = :field
88
+ @record[:field] = MARC::ControlField.new(attributes[TAG])
89
+ when LEAD_TAG then @current_element = :leader
90
+ when REC_TAG then @record[:record] = MARC::Record.new
91
+ end
92
+ end
93
+ end
94
+
95
+ def characters(text)
65
96
  case @current_element
66
- when :leader then @record[:record].leader = text
67
- when :field then @record[:field].value << text
68
97
  when :subfield then @record[:subfield].value << text
98
+ when :field then @record[:field].value << text
99
+ when :leader then @record[:leader] << text
69
100
  end
70
101
  end
71
102
 
72
- def end_element_namespace name, prefix = nil, uri = nil
103
+ def end_element_namespace(name, prefix = nil, uri = nil)
73
104
  @current_element = nil
74
- if uri == @ns
105
+ if (uri == @ns) || @ignore_namespace
75
106
  case name.downcase
76
- when 'record' then yield_record
77
- when /(control|data)field/
78
- @record[:record] << @record[:field]
79
- @record[:field] = nil
80
- @current_element = nil if @current_element == :field
81
- when 'subfield'
107
+ when SF_TAG
82
108
  @record[:field].append(@record[:subfield])
83
109
  @record[:subfield] = nil
84
110
  @current_element = nil if @current_element == :subfield
111
+ when DF_TAG, CF_TAG
112
+ @record[:record] << @record[:field]
113
+ @record[:field] = nil
114
+ @current_element = nil if @current_element == :field
115
+ when REC_TAG then yield_record
116
+ when LEAD_TAG
117
+ @record[:record].leader = @record[:leader]
118
+ @record[:leader] = ""
119
+ @current_element = nil if @current_element == :leader
85
120
  end
86
121
  end
87
- end
122
+ end
88
123
  end
89
-
90
124
 
91
125
  # NokogiriReader uses the Nokogiri SAX Parser to quickly read
92
126
  # a MARCXML document. Because dynamically subclassing MARC::XMLReader
93
127
  # is a little ugly, we need to recreate all of the SAX event methods
94
- # from Nokogiri::XML::SAX::Document here rather than subclassing.
95
- module NokogiriReader
128
+ # from Nokogiri::XML::SAX::Document here rather than subclassing.
129
+ module NokogiriReader
96
130
  include GenericPullParser
131
+
97
132
  def self.extended(receiver)
98
- require 'nokogiri'
133
+ require "nokogiri"
99
134
  receiver.init
100
135
  end
101
-
136
+
102
137
  # Sets our instance variables for SAX parsing in Nokogiri and parser
103
138
  def init
104
- @record = {:record=>nil,:field=>nil,:subfield=>nil}
105
- @current_element = nil
106
- @ns = "http://www.loc.gov/MARC21/slim"
107
- @parser = Nokogiri::XML::SAX::Parser.new(self)
139
+ super
140
+ @parser = Nokogiri::XML::SAX::Parser.new(self)
108
141
  end
109
-
142
+
110
143
  # Loop through the MARC records in the XML document
111
- def each(&block)
112
- unless block_given?
113
- return self.enum_for(:each)
114
- else
144
+ def each(&block)
145
+ if block
115
146
  @block = block
116
147
  @parser.parse(@handle)
148
+ else
149
+ enum_for(:each)
117
150
  end
118
151
  end
119
152
 
120
153
  def error(evt)
121
154
  raise(XMLParseError, "XML parsing error: #{evt}")
122
155
  end
123
-
124
156
 
125
- def method_missing(methName, *args)
126
- sax_methods = [:xmldecl, :start_document, :end_document, :start_element,
127
- :end_element, :comment, :warning, :error, :cdata_block, :processing_instruction]
128
- unless sax_methods.index(methName)
129
- raise NoMethodError.new("undefined method '#{methName} for #{self}", 'no_meth')
157
+ SAX_METHODS = [:xmldecl, :start_document, :end_document, :start_element,
158
+ :end_element, :comment, :warning, :error, :cdata_block, :processing_instruction]
159
+
160
+ def method_missing(method_name, *args)
161
+ unless SAX_METHODS.include?(method_name)
162
+ raise NoMethodError.new("undefined method '#{method_name} for #{self}", "no_meth")
130
163
  end
131
164
  end
132
-
133
- private
134
-
135
- def attributes_to_hash(attributes)
136
- hash = {}
137
- attributes.each do | att |
138
- hash[att.localname] = att.value
139
- end
140
- hash
141
- end
142
- end
143
165
 
166
+ def respond_to_missing?(method_name, include_private = false)
167
+ SAX_METHODS.include?(method_name) || super
168
+ end
169
+
170
+ private
171
+
172
+ def attributes_to_hash(attributes)
173
+ hash = {}
174
+ attributes.each do |att|
175
+ hash[att.localname] = att.value
176
+ end
177
+ hash
178
+ end
179
+ end
144
180
 
145
-
146
181
  # The REXMLReader is the 'default' parser, since we can at least be
147
182
  # assured that REXML is probably there. It uses REXML's PullParser
148
183
  # to handle larger document sizes without consuming insane amounts of
149
- # memory, but it's still REXML (read: slow), so it's a good idea to
184
+ # memory, but it's still REXML (read: slow), so it's a good idea to
150
185
  # use an alternative parser if available. If you don't know the best
151
186
  # parser available, you can use the MagicReader or set:
152
187
  #
@@ -158,43 +193,44 @@ module MARC
158
193
  #
159
194
  # or
160
195
  #
161
- # reader = MARC::XMLReader.new(fh, :parser=>"magic")
196
+ # reader = MARC::XMLReader.new(fh, :parser=>"magic")
162
197
  # (or the constant)
163
198
  #
164
199
  # which will cascade down to REXML if nothing better is found.
165
- #
200
+ #
166
201
  module REXMLReader
167
202
  def self.extended(receiver)
168
- require 'rexml/document'
169
- require 'rexml/parsers/pullparser'
203
+ require "rexml/document"
204
+ require "rexml/parsers/pullparser"
170
205
  receiver.init
171
206
  end
172
-
207
+
173
208
  # Sets our parser
174
209
  def init
175
210
  @parser = REXML::Parsers::PullParser.new(@handle)
176
211
  end
177
-
212
+
178
213
  # Loop through the MARC records in the XML document
179
214
  def each
180
- unless block_given?
181
- return self.enum_for(:each)
182
- else
215
+ if block_given?
183
216
  while @parser.has_next?
184
217
  event = @parser.pull
185
- # if it's the start of a record element
186
- if event.start_element? and strip_ns(event[0]) == 'record'
218
+ # if it's the start of a record element
219
+ if event.start_element? && (strip_ns(event[0]) == "record")
187
220
  yield build_record
188
221
  end
189
- end
222
+ end
223
+ else
224
+ enum_for(:each)
190
225
  end
191
226
  end
192
-
227
+
193
228
  private
229
+
194
230
  def strip_ns(str)
195
- return str.sub(/^.*:/, '')
231
+ str.sub(/^.*:/, "")
196
232
  end
197
-
233
+
198
234
  # will accept parse events until a record has been built up
199
235
  #
200
236
  def build_record
@@ -202,15 +238,15 @@ module MARC
202
238
  data_field = nil
203
239
  control_field = nil
204
240
  subfield = nil
205
- text = ''
241
+ text = ""
206
242
  attrs = nil
207
- if Module.constants.index('Nokogiri') and @parser.is_a?(Nokogiri::XML::Reader)
243
+ if Module.constants.index("Nokogiri") && @parser.is_a?(Nokogiri::XML::Reader)
208
244
  datafield = nil
209
245
  cursor = nil
210
246
  open_elements = []
211
- @parser.each do | node |
247
+ @parser.each do |node|
212
248
  if node.value? && cursor
213
- if cursor.is_a?(Symbol) and cursor == :leader
249
+ if cursor.is_a?(Symbol) && (cursor == :leader)
214
250
  record.leader = node.value
215
251
  else
216
252
  cursor.value = node.value
@@ -230,195 +266,184 @@ module MARC
230
266
  when "controlfield"
231
267
  record << datafield if datafield
232
268
  datafield = nil
233
- control_field = MARC::ControlField.new(node.attribute('tag'))
269
+ control_field = MARC::ControlField.new(node.attribute("tag"))
234
270
  record << control_field
235
271
  cursor = control_field
236
- when "datafield"
272
+ when "datafield"
237
273
  record << datafield if datafield
238
274
  datafield = nil
239
- data_field = MARC::DataField.new(node.attribute('tag'), node.attribute('ind1'), node.attribute('ind2'))
275
+ data_field = MARC::DataField.new(node.attribute("tag"), node.attribute(IND1), node.attribute(IND2))
240
276
  datafield = data_field
241
277
  when "subfield"
242
278
  raise "No datafield to add to" unless datafield
243
- subfield = MARC::Subfield.new(node.attribute('code'))
279
+ subfield = MARC::Subfield.new(node.attribute(CODE))
244
280
  datafield.append(subfield)
245
281
  cursor = subfield
246
282
  when "record"
247
283
  record << datafield if datafield
248
284
  return record
249
- end
250
- #puts node.name
285
+ end
251
286
  end
252
-
287
+
253
288
  else
254
289
  while @parser.has_next?
255
290
  event = @parser.pull
256
291
 
257
292
  if event.text?
258
- text += REXML::Text::unnormalize(event[0])
293
+ text += REXML::Text.unnormalize(event[0])
259
294
  next
260
295
  end
261
296
 
262
297
  if event.start_element?
263
- text = ''
298
+ text = ""
264
299
  attrs = event[1]
265
300
  case strip_ns(event[0])
266
- when 'controlfield'
267
- text = ''
268
- control_field = MARC::ControlField.new(attrs['tag'])
269
- when 'datafield'
270
- text = ''
271
- data_field = MARC::DataField.new(attrs['tag'], attrs['ind1'],
272
- attrs['ind2'])
273
- when 'subfield'
274
- text = ''
275
- subfield = MARC::Subfield.new(attrs['code'])
301
+ when "controlfield"
302
+ text = ""
303
+ control_field = MARC::ControlField.new(attrs[TAG])
304
+ when "datafield"
305
+ text = ""
306
+ data_field = MARC::DataField.new(attrs[TAG], attrs[IND1],
307
+ attrs[IND2])
308
+ when "subfield"
309
+ text = ""
310
+ subfield = MARC::Subfield.new(attrs[CODE])
276
311
  end
277
312
  end
278
313
 
279
314
  if event.end_element?
280
315
  case strip_ns(event[0])
281
- when 'leader'
316
+ when "leader"
282
317
  record.leader = text
283
- when 'record'
318
+ when "record"
284
319
  return record
285
- when 'controlfield'
320
+ when "controlfield"
286
321
  control_field.value = text
287
322
  record.append(control_field)
288
- when 'datafield'
323
+ when "datafield"
289
324
  record.append(data_field)
290
- when 'subfield'
325
+ when "subfield"
291
326
  subfield.value = text
292
327
  data_field.append(subfield)
293
328
  end
294
329
  end
295
330
  end
296
331
  end
297
- end
298
- end
299
-
300
- # The JREXMLReader is really just here to set the load order for
301
- # injecting the Java pull parser.
302
- module JREXMLReader
303
-
304
- def self.extended(receiver)
305
- require 'rexml/document'
306
- require 'rexml/parsers/pullparser'
307
- require 'jrexml'
308
- receiver.extend(REXMLReader)
309
332
  end
310
333
  end
311
-
312
-
313
-
314
-
334
+
315
335
  unless defined? JRUBY_VERSION
316
336
  module LibXMLReader
337
+ def self.extended(receiver)
338
+ require "xml"
339
+ receiver.init
340
+ end
317
341
 
318
- def self.extended(receiver)
319
- require 'xml'
320
- receiver.init
321
- end
322
-
323
- def init
324
- @ns = "http://www.loc.gov/MARC21/slim"
325
- @parser = XML::Reader.io(@handle)
326
- end
342
+ def init
343
+ @ns = "http://www.loc.gov/MARC21/slim"
344
+ @parser = XML::Reader.io(@handle)
345
+ end
327
346
 
328
- def each
329
- unless block_given?
330
- return self.enum_for(:each)
331
- else
332
- while (@parser.read) do
333
- if @parser.local_name == 'record' && @parser.namespace_uri == @ns
334
- yield build_record
335
- end
336
- end # while
347
+ def each
348
+ if block_given?
349
+ while @parser.read
350
+ if @parser.local_name == "record" && @parser.namespace_uri == @ns
351
+ yield build_record
352
+ end
353
+ end # while
354
+ else
355
+ enum_for(:each)
356
+ end
337
357
  end
338
- end # each
339
358
 
340
- def build_record
341
- r = MARC::Record.new()
342
- until (@parser.local_name == 'record' and @parser.node_type == XML::Reader::TYPE_END_ELEMENT) do
359
+ # each
360
+
361
+ def build_record
362
+ r = MARC::Record.new
363
+ until (@parser.local_name == "record") && (@parser.node_type == XML::Reader::TYPE_END_ELEMENT)
343
364
  @parser.read
344
365
  next if @parser.node_type == XML::Reader::TYPE_END_ELEMENT
345
366
  case @parser.local_name
346
- when 'leader'
347
- @parser.read
348
- r.leader = @parser.value
349
- when 'controlfield'
350
- tag = @parser['tag']
351
- @parser.read
352
- r << MARC::ControlField.new(tag, @parser.value)
353
- when 'datafield'
354
- data = MARC::DataField.new(@parser['tag'], @parser['ind1'], @parser['ind2'])
355
- while (@parser.read and !(@parser.local_name == 'datafield' and @parser.node_type == XML::Reader::TYPE_END_ELEMENT)) do
356
- next if @parser.node_type == XML::Reader::TYPE_END_ELEMENT
357
- case @parser.local_name
358
- when 'subfield'
359
- code =@parser['code']
360
- @parser.read
361
- data.append(MARC::Subfield.new(code, @parser.value))
362
- end
363
- end
364
- r << data
367
+ when "leader"
368
+ @parser.read
369
+ r.leader = @parser.value
370
+ when "controlfield"
371
+ tag = @parser[TAG]
372
+ @parser.read
373
+ r << MARC::ControlField.new(tag, @parser.value)
374
+ when "datafield"
375
+ data = MARC::DataField.new(@parser[TAG], @parser[IND1], @parser[IND2])
376
+ while @parser.read && !((@parser.local_name == "datafield") && (@parser.node_type == XML::Reader::TYPE_END_ELEMENT))
377
+ next if @parser.node_type == XML::Reader::TYPE_END_ELEMENT
378
+ case @parser.local_name
379
+ when "subfield"
380
+ code = @parser[CODE]
381
+ @parser.read
382
+ data.append(MARC::Subfield.new(code, @parser.value))
383
+ end
384
+ end
385
+ r << data
365
386
 
366
387
  end # case
367
- end #until
368
- return r
388
+ end # until
389
+ r
390
+ end
369
391
  end
370
392
  end
371
- end
372
393
 
373
394
  # The JrubySTAXReader uses native java calls to parse the incoming stream
374
395
  # of marc-xml. It includes most of the work from GenericPullParser
375
396
 
376
397
  if defined? JRUBY_VERSION
398
+ # *DEPRECATED*: JRubySTAXReader is deprecated and will be removed in a
399
+ # future version of ruby-marc. Please use NokogiriReader
400
+ # instead.
377
401
  module JRubySTAXReader
378
402
  include GenericPullParser
403
+
379
404
  def self.extended(receiver)
380
- require 'java' # may only be neccesary in jruby 1.6
405
+ require "java" # may only be neccesary in jruby 1.6
381
406
  receiver.init
382
407
  end
383
408
 
384
409
  def init
385
- @record = {:record=>nil,:field=>nil,:subfield=>nil}
386
- @current_element = nil
387
- @ns = "http://www.loc.gov/MARC21/slim"
410
+ warn "JRubySTAXReader is deprecated and will be removed in a future version of ruby-marc."
411
+
412
+ super
388
413
  @factory = javax.xml.stream.XMLInputFactory.newInstance
389
414
  @parser = @factory.createXMLStreamReader(@handle.to_inputstream)
390
415
  end
391
416
 
392
417
  # Loop through the MARC records in the XML document
393
- def each(&block)
394
- unless block_given?
395
- return self.enum_for(:each)
396
- else
418
+ def each(&block)
419
+ if block
397
420
  @block = block
398
421
  parser_dispatch
422
+ else
423
+ enum_for(:each)
399
424
  end
400
425
  end
401
426
 
402
427
  def parser_dispatch
403
- while event = @parser.next and event != javax.xml.stream.XMLStreamConstants.END_DOCUMENT do
428
+ while (event = @parser.next) && (event != javax.xml.stream.XMLStreamConstants::END_DOCUMENT)
404
429
  case event
405
- when javax.xml.stream.XMLStreamConstants.START_ELEMENT
406
- start_element_namespace(@parser.getLocalName, [], nil, @parser.getNamespaceURI, nil)
407
- when javax.xml.stream.XMLStreamConstants.END_ELEMENT
408
- end_element_namespace(@parser.getLocalName, @parser.getPrefix, @parser.getNamespaceURI)
409
- when javax.xml.stream.XMLStreamConstants.CHARACTERS
410
- characters(@parser.getText)
430
+ when javax.xml.stream.XMLStreamConstants::START_ELEMENT
431
+ start_element_namespace(@parser.getLocalName, [], nil, @parser.getNamespaceURI, nil)
432
+ when javax.xml.stream.XMLStreamConstants::END_ELEMENT
433
+ end_element_namespace(@parser.getLocalName, @parser.getPrefix, @parser.getNamespaceURI)
434
+ when javax.xml.stream.XMLStreamConstants::CHARACTERS
435
+ characters(@parser.getText)
411
436
  end
412
437
  end
413
438
  end
414
439
 
415
440
  def attributes_to_hash(attributes)
416
441
  hash = {}
417
- @parser.getAttributeCount.times do | i |
442
+ @parser.getAttributeCount.times do |i|
418
443
  hash[@parser.getAttributeName(i).getLocalPart] = @parser.getAttributeValue(i)
419
444
  end
420
445
  hash
421
- end
446
+ end
422
447
  end # end of module
423
- end # end of if jruby
448
+ end # end of if jruby
424
449
  end