libxml-ruby 2.8.0 → 2.9.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (81) hide show
  1. checksums.yaml +4 -4
  2. data/HISTORY +15 -0
  3. data/README.rdoc +7 -7
  4. data/Rakefile +80 -78
  5. data/ext/libxml/extconf.h +4 -0
  6. data/ext/libxml/extconf.rb +57 -116
  7. data/ext/libxml/libxml.c +4 -0
  8. data/ext/libxml/ruby_xml.c +977 -893
  9. data/ext/libxml/ruby_xml.h +20 -10
  10. data/ext/libxml/ruby_xml_attr.c +333 -333
  11. data/ext/libxml/ruby_xml_attr_decl.c +2 -2
  12. data/ext/libxml/ruby_xml_cbg.c +85 -85
  13. data/ext/libxml/ruby_xml_document.c +1133 -1147
  14. data/ext/libxml/ruby_xml_dtd.c +261 -268
  15. data/ext/libxml/ruby_xml_encoding.c +262 -260
  16. data/ext/libxml/ruby_xml_encoding.h +19 -19
  17. data/ext/libxml/ruby_xml_html_parser_context.c +337 -338
  18. data/ext/libxml/ruby_xml_input_cbg.c +191 -191
  19. data/ext/libxml/ruby_xml_io.c +52 -50
  20. data/ext/libxml/ruby_xml_namespace.c +2 -2
  21. data/ext/libxml/ruby_xml_node.c +1446 -1452
  22. data/ext/libxml/ruby_xml_parser_context.c +999 -1001
  23. data/ext/libxml/ruby_xml_reader.c +1226 -1228
  24. data/ext/libxml/ruby_xml_relaxng.c +110 -111
  25. data/ext/libxml/ruby_xml_sax2_handler.c +326 -328
  26. data/ext/libxml/ruby_xml_schema.c +300 -301
  27. data/ext/libxml/ruby_xml_version.h +3 -3
  28. data/ext/libxml/ruby_xml_writer.c +14 -15
  29. data/ext/libxml/ruby_xml_xpath.c +188 -188
  30. data/ext/libxml/ruby_xml_xpath_context.c +360 -361
  31. data/ext/libxml/ruby_xml_xpath_object.c +335 -335
  32. data/libxml-ruby.gemspec +47 -44
  33. data/test/tc_attr.rb +5 -7
  34. data/test/tc_attr_decl.rb +5 -6
  35. data/test/tc_attributes.rb +1 -2
  36. data/test/tc_canonicalize.rb +1 -2
  37. data/test/tc_deprecated_require.rb +1 -2
  38. data/test/tc_document.rb +4 -5
  39. data/test/tc_document_write.rb +2 -3
  40. data/test/tc_dtd.rb +4 -5
  41. data/test/tc_encoding.rb +126 -126
  42. data/test/tc_encoding_sax.rb +4 -3
  43. data/test/tc_error.rb +14 -15
  44. data/test/tc_html_parser.rb +15 -7
  45. data/test/tc_html_parser_context.rb +1 -2
  46. data/test/tc_namespace.rb +2 -3
  47. data/test/tc_namespaces.rb +5 -6
  48. data/test/tc_node.rb +2 -3
  49. data/test/tc_node_cdata.rb +2 -3
  50. data/test/tc_node_comment.rb +1 -2
  51. data/test/tc_node_copy.rb +1 -2
  52. data/test/tc_node_edit.rb +5 -7
  53. data/test/tc_node_pi.rb +1 -2
  54. data/test/tc_node_text.rb +2 -3
  55. data/test/tc_node_write.rb +2 -3
  56. data/test/tc_node_xlink.rb +1 -2
  57. data/test/tc_parser.rb +18 -24
  58. data/test/tc_parser_context.rb +6 -7
  59. data/test/tc_properties.rb +1 -2
  60. data/test/tc_reader.rb +9 -10
  61. data/test/tc_relaxng.rb +4 -5
  62. data/test/tc_sax_parser.rb +9 -10
  63. data/test/tc_schema.rb +4 -5
  64. data/test/tc_traversal.rb +1 -2
  65. data/test/tc_writer.rb +1 -2
  66. data/test/tc_xinclude.rb +1 -2
  67. data/test/tc_xml.rb +1 -2
  68. data/test/tc_xpath.rb +8 -9
  69. data/test/tc_xpath_context.rb +3 -4
  70. data/test/tc_xpath_expression.rb +3 -4
  71. data/test/tc_xpointer.rb +1 -3
  72. data/test/test_helper.rb +3 -1
  73. data/test/test_suite.rb +0 -1
  74. metadata +47 -11
  75. data/test/etc_doc_to_s.rb +0 -21
  76. data/test/ets_doc_file.rb +0 -17
  77. data/test/ets_doc_to_s.rb +0 -23
  78. data/test/ets_gpx.rb +0 -28
  79. data/test/ets_node_gc.rb +0 -23
  80. data/test/ets_test.xml +0 -2
  81. data/test/ets_tsr.rb +0 -11
@@ -1,111 +1,110 @@
1
- #include "ruby_libxml.h"
2
- #include "ruby_xml_relaxng.h"
3
-
4
- /*
5
- * Document-class: LibXML::XML::RelaxNG
6
- *
7
- * The XML::RelaxNG class is used to prepare RelaxNG schemas for validation
8
- * of xml documents.
9
- *
10
- * Schemas can be created from XML documents, strings or URIs using the
11
- * corresponding methods (new for URIs).
12
- *
13
- * Once a schema is prepared, an XML document can be validated by the
14
- * XML::Document#validate_relaxng method providing the XML::RelaxNG object
15
- * as parameter. The method will raise an exception if the document is
16
- * not valid.
17
- *
18
- * Basic Usage:
19
- *
20
- * # parse schema as xml document
21
- * relaxng_document = XML::Document.file('schema.rng')
22
- *
23
- * # prepare schema for validation
24
- * relaxng_schema = XML::RelaxNG.document(relaxng_document)
25
- *
26
- * # parse xml document to be validated
27
- * instance = XML::Document.file('instance.xml')
28
- *
29
- * # validate
30
- * instance.validate_relaxng(relaxng_schema)
31
- */
32
-
33
- VALUE cXMLRelaxNG;
34
-
35
- static void rxml_relaxng_free(xmlRelaxNGPtr xrelaxng)
36
- {
37
- xmlRelaxNGFree(xrelaxng);
38
- }
39
-
40
- /*
41
- * call-seq:
42
- * XML::Relaxng.new(relaxng_uri) -> relaxng
43
- *
44
- * Create a new relaxng from the specified URI.
45
- */
46
- static VALUE rxml_relaxng_init_from_uri(VALUE class, VALUE uri)
47
- {
48
- xmlRelaxNGParserCtxtPtr xparser;
49
- xmlRelaxNGPtr xrelaxng;
50
-
51
- Check_Type(uri, T_STRING);
52
-
53
- xparser = xmlRelaxNGNewParserCtxt(StringValuePtr(uri));
54
- xrelaxng = xmlRelaxNGParse(xparser);
55
- xmlRelaxNGFreeParserCtxt(xparser);
56
-
57
- return Data_Wrap_Struct(cXMLRelaxNG, NULL, rxml_relaxng_free, xrelaxng);
58
- }
59
-
60
- /*
61
- * call-seq:
62
- * XML::RelaxNG.document(document) -> relaxng
63
- *
64
- * Create a new relaxng from the specified document.
65
- */
66
- static VALUE rxml_relaxng_init_from_document(VALUE class, VALUE document)
67
- {
68
- xmlDocPtr xdoc;
69
- xmlRelaxNGPtr xrelaxng;
70
- xmlRelaxNGParserCtxtPtr xparser;
71
-
72
- Data_Get_Struct(document, xmlDoc, xdoc);
73
-
74
- xparser = xmlRelaxNGNewDocParserCtxt(xdoc);
75
- xrelaxng = xmlRelaxNGParse(xparser);
76
- xmlRelaxNGFreeParserCtxt(xparser);
77
-
78
- return Data_Wrap_Struct(cXMLRelaxNG, NULL, rxml_relaxng_free, xrelaxng);
79
- }
80
-
81
- /*
82
- * call-seq:
83
- * XML::RelaxNG.string("relaxng_data") -> "value"
84
- *
85
- * Create a new relaxng using the specified string.
86
- */
87
- static VALUE rxml_relaxng_init_from_string(VALUE self, VALUE relaxng_str)
88
- {
89
- xmlRelaxNGParserCtxtPtr xparser;
90
- xmlRelaxNGPtr xrelaxng;
91
-
92
- Check_Type(relaxng_str, T_STRING);
93
-
94
- xparser = xmlRelaxNGNewMemParserCtxt(StringValuePtr(relaxng_str), strlen(
95
- StringValuePtr(relaxng_str)));
96
- xrelaxng = xmlRelaxNGParse(xparser);
97
- xmlRelaxNGFreeParserCtxt(xparser);
98
-
99
- return Data_Wrap_Struct(cXMLRelaxNG, NULL, rxml_relaxng_free, xrelaxng);
100
- }
101
-
102
- void rxml_init_relaxng(void)
103
- {
104
- cXMLRelaxNG = rb_define_class_under(mXML, "RelaxNG", rb_cObject);
105
- rb_define_singleton_method(cXMLRelaxNG, "new", rxml_relaxng_init_from_uri, 1);
106
- rb_define_singleton_method(cXMLRelaxNG, "from_string",
107
- rxml_relaxng_init_from_string, 1);
108
- rb_define_singleton_method(cXMLRelaxNG, "document",
109
- rxml_relaxng_init_from_document, 1);
110
- }
111
-
1
+ #include "ruby_libxml.h"
2
+ #include "ruby_xml_relaxng.h"
3
+
4
+ /*
5
+ * Document-class: LibXML::XML::RelaxNG
6
+ *
7
+ * The XML::RelaxNG class is used to prepare RelaxNG schemas for validation
8
+ * of xml documents.
9
+ *
10
+ * Schemas can be created from XML documents, strings or URIs using the
11
+ * corresponding methods (new for URIs).
12
+ *
13
+ * Once a schema is prepared, an XML document can be validated by the
14
+ * XML::Document#validate_relaxng method providing the XML::RelaxNG object
15
+ * as parameter. The method will raise an exception if the document is
16
+ * not valid.
17
+ *
18
+ * Basic Usage:
19
+ *
20
+ * # parse schema as xml document
21
+ * relaxng_document = XML::Document.file('schema.rng')
22
+ *
23
+ * # prepare schema for validation
24
+ * relaxng_schema = XML::RelaxNG.document(relaxng_document)
25
+ *
26
+ * # parse xml document to be validated
27
+ * instance = XML::Document.file('instance.xml')
28
+ *
29
+ * # validate
30
+ * instance.validate_relaxng(relaxng_schema)
31
+ */
32
+
33
+ VALUE cXMLRelaxNG;
34
+
35
+ static void rxml_relaxng_free(xmlRelaxNGPtr xrelaxng)
36
+ {
37
+ xmlRelaxNGFree(xrelaxng);
38
+ }
39
+
40
+ /*
41
+ * call-seq:
42
+ * XML::Relaxng.new(relaxng_uri) -> relaxng
43
+ *
44
+ * Create a new relaxng from the specified URI.
45
+ */
46
+ static VALUE rxml_relaxng_init_from_uri(VALUE class, VALUE uri)
47
+ {
48
+ xmlRelaxNGParserCtxtPtr xparser;
49
+ xmlRelaxNGPtr xrelaxng;
50
+
51
+ Check_Type(uri, T_STRING);
52
+
53
+ xparser = xmlRelaxNGNewParserCtxt(StringValuePtr(uri));
54
+ xrelaxng = xmlRelaxNGParse(xparser);
55
+ xmlRelaxNGFreeParserCtxt(xparser);
56
+
57
+ return Data_Wrap_Struct(cXMLRelaxNG, NULL, rxml_relaxng_free, xrelaxng);
58
+ }
59
+
60
+ /*
61
+ * call-seq:
62
+ * XML::RelaxNG.document(document) -> relaxng
63
+ *
64
+ * Create a new relaxng from the specified document.
65
+ */
66
+ static VALUE rxml_relaxng_init_from_document(VALUE class, VALUE document)
67
+ {
68
+ xmlDocPtr xdoc;
69
+ xmlRelaxNGPtr xrelaxng;
70
+ xmlRelaxNGParserCtxtPtr xparser;
71
+
72
+ Data_Get_Struct(document, xmlDoc, xdoc);
73
+
74
+ xparser = xmlRelaxNGNewDocParserCtxt(xdoc);
75
+ xrelaxng = xmlRelaxNGParse(xparser);
76
+ xmlRelaxNGFreeParserCtxt(xparser);
77
+
78
+ return Data_Wrap_Struct(cXMLRelaxNG, NULL, rxml_relaxng_free, xrelaxng);
79
+ }
80
+
81
+ /*
82
+ * call-seq:
83
+ * XML::RelaxNG.string("relaxng_data") -> "value"
84
+ *
85
+ * Create a new relaxng using the specified string.
86
+ */
87
+ static VALUE rxml_relaxng_init_from_string(VALUE self, VALUE relaxng_str)
88
+ {
89
+ xmlRelaxNGParserCtxtPtr xparser;
90
+ xmlRelaxNGPtr xrelaxng;
91
+
92
+ Check_Type(relaxng_str, T_STRING);
93
+
94
+ xparser = xmlRelaxNGNewMemParserCtxt(StringValuePtr(relaxng_str), (int)strlen(StringValuePtr(relaxng_str)));
95
+ xrelaxng = xmlRelaxNGParse(xparser);
96
+ xmlRelaxNGFreeParserCtxt(xparser);
97
+
98
+ return Data_Wrap_Struct(cXMLRelaxNG, NULL, rxml_relaxng_free, xrelaxng);
99
+ }
100
+
101
+ void rxml_init_relaxng(void)
102
+ {
103
+ cXMLRelaxNG = rb_define_class_under(mXML, "RelaxNG", rb_cObject);
104
+ rb_define_singleton_method(cXMLRelaxNG, "new", rxml_relaxng_init_from_uri, 1);
105
+ rb_define_singleton_method(cXMLRelaxNG, "from_string",
106
+ rxml_relaxng_init_from_string, 1);
107
+ rb_define_singleton_method(cXMLRelaxNG, "document",
108
+ rxml_relaxng_init_from_document, 1);
109
+ }
110
+
@@ -1,328 +1,326 @@
1
- /* Please see the LICENSE file for copyright and distribution information */
2
-
3
- #include "ruby_libxml.h"
4
- #include "ruby_xml_sax2_handler.h"
5
-
6
-
7
- VALUE cbidOnCdataBlock;
8
- VALUE cbidOnCharacters;
9
- VALUE cbidOnComment;
10
- VALUE cbidOnEndDocument;
11
- VALUE cbidOnEndElement;
12
- VALUE cbidOnEndElementNs;
13
- VALUE cbidOnExternalSubset;
14
- VALUE cbidOnHasExternalSubset;
15
- VALUE cbidOnHasInternalSubset;
16
- VALUE cbidOnInternalSubset;
17
- VALUE cbidOnIsStandalone;
18
- VALUE cbidOnError;
19
- VALUE cbidOnProcessingInstruction;
20
- VALUE cbidOnReference;
21
- VALUE cbidOnStartElement;
22
- VALUE cbidOnStartElementNs;
23
- VALUE cbidOnStartDocument;
24
-
25
- /* ====== Callbacks =========== */
26
- static void cdata_block_callback(void *ctx,
27
- const char *value, int len)
28
- {
29
- VALUE handler = (VALUE) ctx;
30
-
31
- if (handler != Qnil)
32
- {
33
- rb_funcall(handler, cbidOnCdataBlock,1, rxml_new_cstr_len(value, len, NULL));
34
- }
35
- }
36
-
37
- static void characters_callback(void *ctx, const char *chars, int len)
38
- {
39
- VALUE handler = (VALUE) ctx;
40
-
41
- if (handler != Qnil)
42
- {
43
- VALUE rchars = rxml_new_cstr_len(chars, len, NULL);
44
- rb_funcall(handler, cbidOnCharacters, 1, rchars);
45
- }
46
- }
47
-
48
- static void comment_callback(void *ctx, const char *msg)
49
- {
50
- VALUE handler = (VALUE) ctx;
51
-
52
- if (handler != Qnil)
53
- {
54
- rb_funcall(handler, cbidOnComment,1,rxml_new_cstr(msg, NULL));
55
- }
56
- }
57
-
58
- static void end_document_callback(void *ctx)
59
- {
60
- VALUE handler = (VALUE) ctx;
61
-
62
- if (handler != Qnil)
63
- {
64
- rb_funcall(handler, cbidOnEndDocument, 0);
65
- }
66
- }
67
-
68
- static void end_element_ns_callback(void *ctx,
69
- const xmlChar *xlocalname, const xmlChar *xprefix, const xmlChar *xURI)
70
- {
71
- VALUE handler = (VALUE) ctx;
72
-
73
- if (handler == Qnil)
74
- return;
75
-
76
- /* Call end element for old-times sake */
77
- if (rb_respond_to(handler, cbidOnEndElement))
78
- {
79
- VALUE name;
80
- if (xprefix)
81
- {
82
- name = rxml_new_cstr(xprefix, NULL);
83
- rb_str_cat2(name, ":");
84
- rb_str_cat2(name, xlocalname);
85
- }
86
- else
87
- {
88
- name = rxml_new_cstr(xlocalname, NULL);
89
- }
90
- rb_funcall(handler, cbidOnEndElement, 1, name);
91
- }
92
-
93
- rb_funcall(handler, cbidOnEndElementNs, 3,
94
- rxml_new_cstr(xlocalname, NULL),
95
- xprefix ? rxml_new_cstr(xprefix, NULL) : Qnil,
96
- xURI ? rxml_new_cstr(xURI, NULL) : Qnil);
97
- }
98
-
99
- static void external_subset_callback(void *ctx, const char *name, const char *extid, const char *sysid)
100
- {
101
- VALUE handler = (VALUE) ctx;
102
-
103
- if (handler != Qnil)
104
- {
105
- VALUE rname = name ? rxml_new_cstr(name, NULL) : Qnil;
106
- VALUE rextid = extid ? rxml_new_cstr(extid, NULL) : Qnil;
107
- VALUE rsysid = sysid ? rxml_new_cstr(sysid, NULL) : Qnil;
108
- rb_funcall(handler, cbidOnExternalSubset, 3, rname, rextid, rsysid);
109
- }
110
- }
111
-
112
- static void has_external_subset_callback(void *ctx)
113
- {
114
- VALUE handler = (VALUE) ctx;
115
-
116
- if (handler != Qnil)
117
- {
118
- rb_funcall(handler, cbidOnHasExternalSubset, 0);
119
- }
120
- }
121
-
122
- static void has_internal_subset_callback(void *ctx)
123
- {
124
- VALUE handler = (VALUE) ctx;
125
-
126
- if (handler != Qnil)
127
- {
128
- rb_funcall(handler, cbidOnHasInternalSubset, 0);
129
- }
130
- }
131
-
132
- static void internal_subset_callback(void *ctx, const char *name, const char *extid, const char *sysid)
133
- {
134
- VALUE handler = (VALUE) ctx;
135
-
136
- if (handler != Qnil)
137
- {
138
- VALUE rname = name ? rxml_new_cstr(name, NULL) : Qnil;
139
- VALUE rextid = extid ? rxml_new_cstr(extid, NULL) : Qnil;
140
- VALUE rsysid = sysid ? rxml_new_cstr(sysid, NULL) : Qnil;
141
- rb_funcall(handler, cbidOnInternalSubset, 3, rname, rextid, rsysid);
142
- }
143
- }
144
-
145
- static void is_standalone_callback(void *ctx)
146
- {
147
- VALUE handler = (VALUE) ctx;
148
-
149
- if (handler != Qnil)
150
- {
151
- rb_funcall(handler, cbidOnIsStandalone,0);
152
- }
153
- }
154
-
155
- static void processing_instruction_callback(void *ctx, const char *target, const char *data)
156
- {
157
- VALUE handler = (VALUE) ctx;
158
-
159
- if (handler != Qnil)
160
- {
161
- VALUE rtarget = target ? rxml_new_cstr(target, NULL) : Qnil;
162
- VALUE rdata = data ? rxml_new_cstr(data, NULL) : Qnil;
163
- rb_funcall(handler, cbidOnProcessingInstruction, 2, rtarget, rdata);
164
- }
165
- }
166
-
167
- static void reference_callback(void *ctx, const char *name)
168
- {
169
- VALUE handler = (VALUE) ctx;
170
-
171
- if (handler != Qnil)
172
- {
173
- rb_funcall(handler, cbidOnReference,1,rxml_new_cstr(name, NULL));
174
- }
175
- }
176
-
177
- static void start_document_callback(void *ctx)
178
- {
179
- VALUE handler = (VALUE) ctx;
180
-
181
- if (handler != Qnil)
182
- {
183
- rb_funcall(handler, cbidOnStartDocument, 0);
184
- }
185
- }
186
-
187
- static void start_element_ns_callback(void *ctx,
188
- const xmlChar *xlocalname, const xmlChar *xprefix, const xmlChar *xURI,
189
- int nb_namespaces, const xmlChar **xnamespaces,
190
- int nb_attributes, int nb_defaulted, const xmlChar **xattributes)
191
- {
192
- VALUE handler = (VALUE) ctx;
193
- VALUE attributes = rb_hash_new();
194
- VALUE namespaces = rb_hash_new();
195
-
196
- if (handler == Qnil)
197
- return;
198
-
199
- if (xattributes)
200
- {
201
- /* Each attribute is an array of [localname, prefix, URI, value, end] */
202
- int i;
203
- for (i = 0;i < nb_attributes * 5; i+=5)
204
- {
205
- VALUE attrName = rxml_new_cstr(xattributes[i+0], NULL);
206
- VALUE attrValue = rxml_new_cstr_len(xattributes[i+3], xattributes[i+4] - xattributes[i+3], NULL);
207
-
208
- rb_hash_aset(attributes, attrName, attrValue);
209
- }
210
- }
211
-
212
- if (xnamespaces)
213
- {
214
- int i;
215
- for (i = 0;i < nb_namespaces * 2; i+=2)
216
- {
217
- VALUE nsPrefix = xnamespaces[i+0] ? rxml_new_cstr(xnamespaces[i+0], NULL) : Qnil;
218
- VALUE nsURI = xnamespaces[i+1] ? rxml_new_cstr(xnamespaces[i+1], NULL) : Qnil;
219
- rb_hash_aset(namespaces, nsPrefix, nsURI);
220
- }
221
- }
222
-
223
- /* Call start element for old-times sake */
224
- if (rb_respond_to(handler, cbidOnStartElement))
225
- {
226
- VALUE name;
227
- if (xprefix)
228
- {
229
- name = rxml_new_cstr(xprefix, NULL);
230
- rb_str_cat2(name, ":");
231
- rb_str_cat2(name, xlocalname);
232
- }
233
- else
234
- {
235
- name = rxml_new_cstr(xlocalname, NULL);
236
- }
237
- rb_funcall(handler, cbidOnStartElement, 2, name, attributes);
238
- }
239
-
240
- rb_funcall(handler, cbidOnStartElementNs, 5,
241
- rxml_new_cstr(xlocalname, NULL),
242
- attributes,
243
- xprefix ? rxml_new_cstr(xprefix, NULL) : Qnil,
244
- xURI ? rxml_new_cstr(xURI, NULL) : Qnil,
245
- namespaces);
246
- }
247
-
248
- static void structured_error_callback(void *ctx, xmlErrorPtr xerror)
249
- {
250
- /* Older versions of Libxml will pass a NULL context from the sax parser. Fixed on
251
- Feb 23, 2011. See:
252
-
253
- http://git.gnome.org/browse/libxml2/commit/?id=241d4a1069e6bedd0ee2295d7b43858109c1c6d1 */
254
-
255
- VALUE handler;
256
-
257
- #if LIBXML_VERSION <= 20708
258
- xmlParserCtxtPtr ctxt = (xmlParserCtxt*)(xerror->ctxt);
259
- ctx = ctxt->userData;
260
- #endif
261
-
262
- handler = (VALUE) ctx;
263
-
264
- if (handler != Qnil)
265
- {
266
- VALUE error = rxml_error_wrap(xerror);
267
- rb_funcall(handler, cbidOnError, 1, error);
268
- }
269
- }
270
-
271
- /* ====== Handler =========== */
272
- xmlSAXHandler rxml_sax_handler = {
273
- (internalSubsetSAXFunc) internal_subset_callback,
274
- (isStandaloneSAXFunc) is_standalone_callback,
275
- (hasInternalSubsetSAXFunc) has_internal_subset_callback,
276
- (hasExternalSubsetSAXFunc) has_external_subset_callback,
277
- 0, /* resolveEntity */
278
- 0, /* getEntity */
279
- 0, /* entityDecl */
280
- 0, /* notationDecl */
281
- 0, /* attributeDecl */
282
- 0, /* elementDecl */
283
- 0, /* unparsedEntityDecl */
284
- 0, /* setDocumentLocator */
285
- (startDocumentSAXFunc) start_document_callback,
286
- (endDocumentSAXFunc) end_document_callback,
287
- 0, /* Use start_element_ns_callback instead */
288
- 0, /* Use end_element_ns_callback instead */
289
- (referenceSAXFunc) reference_callback,
290
- (charactersSAXFunc) characters_callback,
291
- 0, /* ignorableWhitespace */
292
- (processingInstructionSAXFunc) processing_instruction_callback,
293
- (commentSAXFunc) comment_callback,
294
- 0, /* xmlStructuredErrorFunc is used instead */
295
- 0, /* xmlStructuredErrorFunc is used instead */
296
- 0, /* xmlStructuredErrorFunc is used instead */
297
- 0, /* xmlGetParameterEntity */
298
- (cdataBlockSAXFunc) cdata_block_callback,
299
- (externalSubsetSAXFunc) external_subset_callback,
300
- XML_SAX2_MAGIC, /* force SAX2 */
301
- 0, /* _private */
302
- (startElementNsSAX2Func) start_element_ns_callback,
303
- (endElementNsSAX2Func) end_element_ns_callback,
304
- (xmlStructuredErrorFunc) structured_error_callback
305
- };
306
-
307
- void rxml_init_sax2_handler(void)
308
- {
309
-
310
- /* SaxCallbacks */
311
- cbidOnCdataBlock = rb_intern("on_cdata_block");
312
- cbidOnCharacters = rb_intern("on_characters");
313
- cbidOnComment = rb_intern("on_comment");
314
- cbidOnEndDocument = rb_intern("on_end_document");
315
- cbidOnEndElement = rb_intern("on_end_element");
316
- cbidOnEndElementNs = rb_intern("on_end_element_ns");
317
- cbidOnError = rb_intern("on_error");
318
- cbidOnExternalSubset = rb_intern("on_external_subset");
319
- cbidOnHasExternalSubset = rb_intern("on_has_external_subset");
320
- cbidOnHasInternalSubset = rb_intern("on_has_internal_subset");
321
- cbidOnInternalSubset = rb_intern("on_internal_subset");
322
- cbidOnIsStandalone = rb_intern("on_is_standalone");
323
- cbidOnProcessingInstruction = rb_intern("on_processing_instruction");
324
- cbidOnReference = rb_intern("on_reference");
325
- cbidOnStartElement = rb_intern("on_start_element");
326
- cbidOnStartElementNs = rb_intern("on_start_element_ns");
327
- cbidOnStartDocument = rb_intern("on_start_document");
328
- }
1
+ /* Please see the LICENSE file for copyright and distribution information */
2
+
3
+ #include "ruby_libxml.h"
4
+ #include "ruby_xml_sax2_handler.h"
5
+
6
+
7
+ VALUE cbidOnCdataBlock;
8
+ VALUE cbidOnCharacters;
9
+ VALUE cbidOnComment;
10
+ VALUE cbidOnEndDocument;
11
+ VALUE cbidOnEndElement;
12
+ VALUE cbidOnEndElementNs;
13
+ VALUE cbidOnExternalSubset;
14
+ VALUE cbidOnHasExternalSubset;
15
+ VALUE cbidOnHasInternalSubset;
16
+ VALUE cbidOnInternalSubset;
17
+ VALUE cbidOnIsStandalone;
18
+ VALUE cbidOnError;
19
+ VALUE cbidOnProcessingInstruction;
20
+ VALUE cbidOnReference;
21
+ VALUE cbidOnStartElement;
22
+ VALUE cbidOnStartElementNs;
23
+ VALUE cbidOnStartDocument;
24
+
25
+ /* ====== Callbacks =========== */
26
+ static void cdata_block_callback(void *ctx, const xmlChar *value, int len)
27
+ {
28
+ VALUE handler = (VALUE) ctx;
29
+
30
+ if (handler != Qnil)
31
+ {
32
+ rb_funcall(handler, cbidOnCdataBlock,1, rxml_new_cstr_len(value, len, NULL));
33
+ }
34
+ }
35
+
36
+ static void characters_callback(void *ctx, const xmlChar *chars, int len)
37
+ {
38
+ VALUE handler = (VALUE) ctx;
39
+
40
+ if (handler != Qnil)
41
+ {
42
+ VALUE rchars = rxml_new_cstr_len(chars, len, NULL);
43
+ rb_funcall(handler, cbidOnCharacters, 1, rchars);
44
+ }
45
+ }
46
+
47
+ static void comment_callback(void *ctx, const xmlChar *msg)
48
+ {
49
+ VALUE handler = (VALUE) ctx;
50
+
51
+ if (handler != Qnil)
52
+ {
53
+ rb_funcall(handler, cbidOnComment, 1, rxml_new_cstr(msg, NULL));
54
+ }
55
+ }
56
+
57
+ static void end_document_callback(void *ctx)
58
+ {
59
+ VALUE handler = (VALUE) ctx;
60
+
61
+ if (handler != Qnil)
62
+ {
63
+ rb_funcall(handler, cbidOnEndDocument, 0);
64
+ }
65
+ }
66
+
67
+ static void end_element_ns_callback(void *ctx, const xmlChar *xlocalname, const xmlChar *xprefix, const xmlChar *xURI)
68
+ {
69
+ VALUE handler = (VALUE) ctx;
70
+
71
+ if (handler == Qnil)
72
+ return;
73
+
74
+ /* Call end element for old-times sake */
75
+ if (rb_respond_to(handler, cbidOnEndElement))
76
+ {
77
+ VALUE name;
78
+ if (xprefix)
79
+ {
80
+ name = rxml_new_cstr(xprefix, NULL);
81
+ rb_str_cat2(name, ":");
82
+ rb_str_cat2(name, (const char*)xlocalname);
83
+ }
84
+ else
85
+ {
86
+ name = rxml_new_cstr(xlocalname, NULL);
87
+ }
88
+ rb_funcall(handler, cbidOnEndElement, 1, name);
89
+ }
90
+
91
+ rb_funcall(handler, cbidOnEndElementNs, 3,
92
+ rxml_new_cstr(xlocalname, NULL),
93
+ xprefix ? rxml_new_cstr(xprefix, NULL) : Qnil,
94
+ xURI ? rxml_new_cstr(xURI, NULL) : Qnil);
95
+ }
96
+
97
+ static void external_subset_callback(void *ctx, const xmlChar *name, const xmlChar *extid, const xmlChar *sysid)
98
+ {
99
+ VALUE handler = (VALUE) ctx;
100
+
101
+ if (handler != Qnil)
102
+ {
103
+ VALUE rname = name ? rxml_new_cstr(name, NULL) : Qnil;
104
+ VALUE rextid = extid ? rxml_new_cstr(extid, NULL) : Qnil;
105
+ VALUE rsysid = sysid ? rxml_new_cstr(sysid, NULL) : Qnil;
106
+ rb_funcall(handler, cbidOnExternalSubset, 3, rname, rextid, rsysid);
107
+ }
108
+ }
109
+
110
+ static void has_external_subset_callback(void *ctx)
111
+ {
112
+ VALUE handler = (VALUE) ctx;
113
+
114
+ if (handler != Qnil)
115
+ {
116
+ rb_funcall(handler, cbidOnHasExternalSubset, 0);
117
+ }
118
+ }
119
+
120
+ static void has_internal_subset_callback(void *ctx)
121
+ {
122
+ VALUE handler = (VALUE) ctx;
123
+
124
+ if (handler != Qnil)
125
+ {
126
+ rb_funcall(handler, cbidOnHasInternalSubset, 0);
127
+ }
128
+ }
129
+
130
+ static void internal_subset_callback(void *ctx, const xmlChar *name, const xmlChar *extid, const xmlChar *sysid)
131
+ {
132
+ VALUE handler = (VALUE) ctx;
133
+
134
+ if (handler != Qnil)
135
+ {
136
+ VALUE rname = name ? rxml_new_cstr(name, NULL) : Qnil;
137
+ VALUE rextid = extid ? rxml_new_cstr(extid, NULL) : Qnil;
138
+ VALUE rsysid = sysid ? rxml_new_cstr(sysid, NULL) : Qnil;
139
+ rb_funcall(handler, cbidOnInternalSubset, 3, rname, rextid, rsysid);
140
+ }
141
+ }
142
+
143
+ static void is_standalone_callback(void *ctx)
144
+ {
145
+ VALUE handler = (VALUE) ctx;
146
+
147
+ if (handler != Qnil)
148
+ {
149
+ rb_funcall(handler, cbidOnIsStandalone,0);
150
+ }
151
+ }
152
+
153
+ static void processing_instruction_callback(void *ctx, const xmlChar *target, const xmlChar *data)
154
+ {
155
+ VALUE handler = (VALUE) ctx;
156
+
157
+ if (handler != Qnil)
158
+ {
159
+ VALUE rtarget = target ? rxml_new_cstr(target, NULL) : Qnil;
160
+ VALUE rdata = data ? rxml_new_cstr(data, NULL) : Qnil;
161
+ rb_funcall(handler, cbidOnProcessingInstruction, 2, rtarget, rdata);
162
+ }
163
+ }
164
+
165
+ static void reference_callback(void *ctx, const xmlChar *name)
166
+ {
167
+ VALUE handler = (VALUE) ctx;
168
+
169
+ if (handler != Qnil)
170
+ {
171
+ rb_funcall(handler, cbidOnReference, 1, rxml_new_cstr(name, NULL));
172
+ }
173
+ }
174
+
175
+ static void start_document_callback(void *ctx)
176
+ {
177
+ VALUE handler = (VALUE) ctx;
178
+
179
+ if (handler != Qnil)
180
+ {
181
+ rb_funcall(handler, cbidOnStartDocument, 0);
182
+ }
183
+ }
184
+
185
+ static void start_element_ns_callback(void *ctx,
186
+ const xmlChar *xlocalname, const xmlChar *xprefix, const xmlChar *xURI,
187
+ int nb_namespaces, const xmlChar **xnamespaces,
188
+ int nb_attributes, int nb_defaulted, const xmlChar **xattributes)
189
+ {
190
+ VALUE handler = (VALUE) ctx;
191
+ VALUE attributes = rb_hash_new();
192
+ VALUE namespaces = rb_hash_new();
193
+
194
+ if (handler == Qnil)
195
+ return;
196
+
197
+ if (xattributes)
198
+ {
199
+ /* Each attribute is an array of [localname, prefix, URI, value, end] */
200
+ int i;
201
+ for (i = 0;i < nb_attributes * 5; i+=5)
202
+ {
203
+ VALUE attrName = rxml_new_cstr(xattributes[i+0], NULL);
204
+ long attrLen = xattributes[i+4] - xattributes[i+3];
205
+ VALUE attrValue = rxml_new_cstr_len(xattributes[i+3], attrLen, NULL);
206
+ rb_hash_aset(attributes, attrName, attrValue);
207
+ }
208
+ }
209
+
210
+ if (xnamespaces)
211
+ {
212
+ int i;
213
+ for (i = 0;i < nb_namespaces * 2; i+=2)
214
+ {
215
+ VALUE nsPrefix = xnamespaces[i+0] ? rxml_new_cstr(xnamespaces[i+0], NULL) : Qnil;
216
+ VALUE nsURI = xnamespaces[i+1] ? rxml_new_cstr(xnamespaces[i+1], NULL) : Qnil;
217
+ rb_hash_aset(namespaces, nsPrefix, nsURI);
218
+ }
219
+ }
220
+
221
+ /* Call start element for old-times sake */
222
+ if (rb_respond_to(handler, cbidOnStartElement))
223
+ {
224
+ VALUE name;
225
+ if (xprefix)
226
+ {
227
+ name = rxml_new_cstr(xprefix, NULL);
228
+ rb_str_cat2(name, ":");
229
+ rb_str_cat2(name, (const char*)xlocalname);
230
+ }
231
+ else
232
+ {
233
+ name = rxml_new_cstr(xlocalname, NULL);
234
+ }
235
+ rb_funcall(handler, cbidOnStartElement, 2, name, attributes);
236
+ }
237
+
238
+ rb_funcall(handler, cbidOnStartElementNs, 5,
239
+ rxml_new_cstr(xlocalname, NULL),
240
+ attributes,
241
+ xprefix ? rxml_new_cstr(xprefix, NULL) : Qnil,
242
+ xURI ? rxml_new_cstr(xURI, NULL) : Qnil,
243
+ namespaces);
244
+ }
245
+
246
+ static void structured_error_callback(void *ctx, xmlErrorPtr xerror)
247
+ {
248
+ /* Older versions of Libxml will pass a NULL context from the sax parser. Fixed on
249
+ Feb 23, 2011. See:
250
+
251
+ http://git.gnome.org/browse/libxml2/commit/?id=241d4a1069e6bedd0ee2295d7b43858109c1c6d1 */
252
+
253
+ VALUE handler;
254
+
255
+ #if LIBXML_VERSION <= 20708
256
+ xmlParserCtxtPtr ctxt = (xmlParserCtxt*)(xerror->ctxt);
257
+ ctx = ctxt->userData;
258
+ #endif
259
+
260
+ handler = (VALUE) ctx;
261
+
262
+ if (handler != Qnil)
263
+ {
264
+ VALUE error = rxml_error_wrap(xerror);
265
+ rb_funcall(handler, cbidOnError, 1, error);
266
+ }
267
+ }
268
+
269
+ /* ====== Handler =========== */
270
+ xmlSAXHandler rxml_sax_handler = {
271
+ (internalSubsetSAXFunc) internal_subset_callback,
272
+ (isStandaloneSAXFunc) is_standalone_callback,
273
+ (hasInternalSubsetSAXFunc) has_internal_subset_callback,
274
+ (hasExternalSubsetSAXFunc) has_external_subset_callback,
275
+ 0, /* resolveEntity */
276
+ 0, /* getEntity */
277
+ 0, /* entityDecl */
278
+ 0, /* notationDecl */
279
+ 0, /* attributeDecl */
280
+ 0, /* elementDecl */
281
+ 0, /* unparsedEntityDecl */
282
+ 0, /* setDocumentLocator */
283
+ (startDocumentSAXFunc) start_document_callback,
284
+ (endDocumentSAXFunc) end_document_callback,
285
+ 0, /* Use start_element_ns_callback instead */
286
+ 0, /* Use end_element_ns_callback instead */
287
+ (referenceSAXFunc) reference_callback,
288
+ (charactersSAXFunc) characters_callback,
289
+ 0, /* ignorableWhitespace */
290
+ (processingInstructionSAXFunc) processing_instruction_callback,
291
+ (commentSAXFunc) comment_callback,
292
+ 0, /* xmlStructuredErrorFunc is used instead */
293
+ 0, /* xmlStructuredErrorFunc is used instead */
294
+ 0, /* xmlStructuredErrorFunc is used instead */
295
+ 0, /* xmlGetParameterEntity */
296
+ (cdataBlockSAXFunc) cdata_block_callback,
297
+ (externalSubsetSAXFunc) external_subset_callback,
298
+ XML_SAX2_MAGIC, /* force SAX2 */
299
+ 0, /* _private */
300
+ (startElementNsSAX2Func) start_element_ns_callback,
301
+ (endElementNsSAX2Func) end_element_ns_callback,
302
+ (xmlStructuredErrorFunc) structured_error_callback
303
+ };
304
+
305
+ void rxml_init_sax2_handler(void)
306
+ {
307
+
308
+ /* SaxCallbacks */
309
+ cbidOnCdataBlock = rb_intern("on_cdata_block");
310
+ cbidOnCharacters = rb_intern("on_characters");
311
+ cbidOnComment = rb_intern("on_comment");
312
+ cbidOnEndDocument = rb_intern("on_end_document");
313
+ cbidOnEndElement = rb_intern("on_end_element");
314
+ cbidOnEndElementNs = rb_intern("on_end_element_ns");
315
+ cbidOnError = rb_intern("on_error");
316
+ cbidOnExternalSubset = rb_intern("on_external_subset");
317
+ cbidOnHasExternalSubset = rb_intern("on_has_external_subset");
318
+ cbidOnHasInternalSubset = rb_intern("on_has_internal_subset");
319
+ cbidOnInternalSubset = rb_intern("on_internal_subset");
320
+ cbidOnIsStandalone = rb_intern("on_is_standalone");
321
+ cbidOnProcessingInstruction = rb_intern("on_processing_instruction");
322
+ cbidOnReference = rb_intern("on_reference");
323
+ cbidOnStartElement = rb_intern("on_start_element");
324
+ cbidOnStartElementNs = rb_intern("on_start_element_ns");
325
+ cbidOnStartDocument = rb_intern("on_start_document");
326
+ }