次のコードを含む XML ファイルがあります。
<Root>
<!DOCTYPE stylesheet [
<!ENTITY CLARK_HISTORICAL_ALLOCATION_CLASS "location.stuff.things">
<!ENTITY CLARK_UNIFORM_ALLOCATION_CLASS "location.stuff.items">
<!ENTITY CLARK_PSEUDO_UNIFORM_ALLOCATION_CLASS "location.items.stuff">
]>
<ServerConfig>
<host name="allen" env="flat"/>
</ServerConfig>
<ClientConfig>
<host name="george" env="flat"/>
<host name="alice" env="flat"/>
<host name="bernice" env="flat"/>
</ClientConfig>
</Root>
次のようにファイル入力ストリームとしてファイルを読み込むことで、DTD を無視し、ファイルの ClientConfig 部分にノードを追加しようとするコードがあります。
val factory = javax.xml.parsers.SAXParserFactory.newInstance()
factory.setValidating(false)
factory.setFeature("http://xml.org/sax/features/validation", false)
factory.setFeature("http://apache.org/xml/features/nonvalidating/load-dtd-grammar", false)
factory.setFeature("http://apache.org/xml/features/nonvalidating/load-external-dtd", false)
factory.setFeature("http://xml.org/sax/features/external-general-entities", false)
factory.setFeature("http://xml.org/sax/features/external-parameter-entities", false)
val data = scala.xml.XML.withSAXParser(factory.newSAXParser).load(FileInputstream)
val p = new XMLPrettyPrinter
val added = addNewEntry(data, "bob", "flat")
def toBeAddedEntry(name: String, env: String) = <host name={ name } env={ env } />
def addNewEntry(originalXML: Elem, name: String, env: String) = {
originalXML match {
case e @ Elem(_, _, _, _, configs @ _*) => {
val changedNodes = configs.map {
case <ClientConfig>{ innerConfigs @ _* }</ClientConfig> => {
<ClientConfig> { toBeAddedEntry(name, env) ++ innerConfigs }</ClientConfig>
}
case other => other
}
e.copy(child = changedNodes)
}
case _ => originalXML
}
}
p.write(added)(System.out)
ただし、これらすべてを追加しても、DTD は XML パーサーによって無視されず、XML ファイルで展開/解決されます。DTD が無視されないのはなぜですか?
また、DTD を無視するために次のリンクをたどったことも追加したいと思います。