mirror of
				https://gitlab.gnome.org/GNOME/libxml2.git
				synced 2025-10-26 00:37:43 +03:00 
			
		
		
		
	Always reported to generic error, not to parser context for backward compatibility. Several downstream test suites rely on this behavior.
		
			
				
	
	
		
			150 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			150 lines
		
	
	
		
			5.1 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python3
 | |
| #
 | |
| # This tests custom input callbacks
 | |
| #
 | |
| import sys
 | |
| import setup_test
 | |
| import libxml2
 | |
| try:
 | |
|     import StringIO
 | |
|     str_io = StringIO.StringIO
 | |
| except:
 | |
|     import io
 | |
|     str_io = io.StringIO
 | |
| 
 | |
| # We implement a new scheme, py://strings/ that will reference this dictionary
 | |
| pystrings = {
 | |
|     'catalogs/catalog.xml' :
 | |
| '''<?xml version="1.0" encoding="utf-8"?>
 | |
| <!DOCTYPE catalog PUBLIC "-//OASIS//DTD Entity Resolution XML Catalog V1.0//EN" "http://www.oasis-open.org/committees/entity/release/1.0/catalog.dtd">
 | |
| <catalog xmlns="urn:oasis:names:tc:entity:xmlns:xml:catalog">
 | |
|   <rewriteSystem systemIdStartString="http://example.com/dtds/" rewritePrefix="../dtds/"/>
 | |
| </catalog>''',
 | |
| 
 | |
|     'xml/sample.xml' :
 | |
| '''<?xml version="1.0" encoding="utf-8"?>
 | |
| <!DOCTYPE root SYSTEM "http://example.com/dtds/sample.dtd">
 | |
| <root>&sample.entity;</root>''',
 | |
| 
 | |
|     'dtds/sample.dtd' :
 | |
| '''
 | |
| <!ELEMENT root (#PCDATA)>
 | |
| <!ENTITY sample.entity "replacement text">'''
 | |
| }
 | |
| 
 | |
| prefix = "py://strings/"
 | |
| startURL = prefix + "xml/sample.xml"
 | |
| catURL = prefix + "catalogs/catalog.xml"
 | |
| 
 | |
| def my_input_cb(URI):
 | |
|     if not(URI.startswith(prefix)):
 | |
|         return None
 | |
|     path = URI[len(prefix):]
 | |
|     if path not in pystrings:
 | |
|         return None
 | |
|     return str_io(pystrings[path])
 | |
| 
 | |
| 
 | |
| def run_test(desc, docpath, catalog, exp_status="verified", exp_err=[], test_callback=None,
 | |
|         root_name="root", root_content="replacement text"):
 | |
|     opts = libxml2.XML_PARSE_DTDLOAD | libxml2.XML_PARSE_NONET | libxml2.XML_PARSE_COMPACT
 | |
|     actual_err = []
 | |
| 
 | |
|     def my_global_error_cb(ctx, msg):
 | |
|         actual_err.append((-1, msg))
 | |
|     def my_ctx_error_cb(arg, msg, severity, reserved):
 | |
|         actual_err.append((severity, msg))
 | |
| 
 | |
|     libxml2.registerErrorHandler(my_global_error_cb, None)
 | |
|     try:
 | |
|         parser = libxml2.createURLParserCtxt(docpath, opts)
 | |
|         parser.setErrorHandler(my_ctx_error_cb, None)
 | |
|         if catalog is not None:
 | |
|             parser.addLocalCatalog(catalog)
 | |
|         if test_callback is not None:
 | |
|             test_callback()
 | |
|         parser.parseDocument()
 | |
|         doc = parser.doc()
 | |
|         actual_status = "loaded"
 | |
|         e = doc.getRootElement()
 | |
|         if e.name == root_name and e.content == root_content:
 | |
|             actual_status = "verified"
 | |
|         doc.freeDoc()
 | |
|     except libxml2.parserError:
 | |
|         actual_status = "not loaded"
 | |
| 
 | |
|     if actual_status != exp_status:
 | |
|         print("Test '%s' failed: expect status '%s', actual '%s'" % (desc, exp_status, actual_status))
 | |
|         sys.exit(1)
 | |
|     elif actual_err != exp_err:
 | |
|         print("Test '%s' failed" % desc)
 | |
|         print("Expect errors:")
 | |
|         for s,m in exp_err: print("  [%2d] '%s'" % (s,m))
 | |
|         print("Actual errors:")
 | |
|         for s,m in actual_err: print("  [%2d] '%s'" % (s,m))
 | |
|         sys.exit(1)
 | |
| 
 | |
| 
 | |
| # Check that we cannot read custom schema without custom callback
 | |
| run_test(desc="Loading entity without custom callback",
 | |
|         docpath=startURL, catalog=None,
 | |
|         exp_status="not loaded", exp_err=[
 | |
|             (-1, "I/O "),
 | |
|             (-1, "warning : "),
 | |
|             (-1, "failed to load \"py://strings/xml/sample.xml\": No such file or directory\n")
 | |
|             ])
 | |
| 
 | |
| # Register handler and try to load the same entity
 | |
| libxml2.registerInputCallback(my_input_cb)
 | |
| run_test(desc="Loading entity with custom callback",
 | |
|         docpath=startURL, catalog=None,
 | |
|         exp_status="loaded", exp_err=[
 | |
|             ( -1, "Attempt to load network entity http://example.com/dtds/sample.dtd"),
 | |
|             (  4, "Entity 'sample.entity' not defined\n")
 | |
|             ])
 | |
| 
 | |
| # Register a catalog (also accessible via pystr://) and retry
 | |
| run_test(desc="Loading entity with custom callback and catalog",
 | |
|         docpath=startURL, catalog=catURL)
 | |
| 
 | |
| # Unregister custom callback when parser is already created
 | |
| run_test(desc="Loading entity and unregistering callback",
 | |
|         docpath=startURL, catalog=catURL,
 | |
|         test_callback=lambda: libxml2.popInputCallbacks(),
 | |
|         exp_status="loaded", exp_err=[
 | |
|             ( 3, "failed to load \"py://strings/dtds/sample.dtd\": No such file or directory\n"),
 | |
|             ( 4, "Entity 'sample.entity' not defined\n")
 | |
|             ])
 | |
| 
 | |
| # Try to load the document again
 | |
| run_test(desc="Retry loading document after unregistering callback",
 | |
|         docpath=startURL, catalog=catURL,
 | |
|         exp_status="not loaded", exp_err=[
 | |
|             (-1, "I/O "),
 | |
|             (-1, "warning : "),
 | |
|             (-1, "failed to load \"py://strings/xml/sample.xml\": No such file or directory\n")
 | |
|             ])
 | |
| 
 | |
| # But should be able to read standard I/O yet...
 | |
| run_test(desc="Loading using standard i/o after unregistering callback",
 | |
|         docpath="tst.xml", catalog=None,
 | |
|         root_name='doc', root_content='bar')
 | |
| 
 | |
| # Now pop ALL input callbacks, should fail to load even standard I/O
 | |
| try:
 | |
|     while True:
 | |
|         libxml2.popInputCallbacks()
 | |
| except IndexError:
 | |
|     pass
 | |
| 
 | |
| run_test(desc="Loading using standard i/o after unregistering all callbacks",
 | |
|         docpath="tst.xml", catalog=None,
 | |
|         exp_status="not loaded", exp_err=[
 | |
|             (-1, "I/O "),
 | |
|             (-1, "warning : "),
 | |
|             (-1, "failed to load \"tst.xml\": No such file or directory\n")
 | |
|             ])
 | |
| 
 | |
| print("OK")
 | |
| sys.exit(0);
 |