libxml-ruby 2.8.0 → 2.9.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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
+ }