0

log4j.xml ファイルを解析し、いくつかの属性を編集して書き戻そうとしています。

log4j.xml に<!DOCTYPE log4j:configuration SYSTEM "log4j.dtd">宣言がありますが、書き戻すと、宣言が に変わりました<!DOCTYPE log4j>

で解析するためにファイルを開き、xmlDoc = Document.new(File.new(file, 'r'))で書きましたxmlDoc.write(File.new(file, 'w'), 0)

で開いてみましたxmlDoc = Document.new(File.new(file, 'r'), { :raw => :all })

元の DOCTYPE 宣言を保持する方法はありますか?

どうもありがとうございました!

4

1 に答える 1

0

使用法では不可能だと思いrexmlます。その簡単な要約を見てください-これはrexmlライブラリで行われるプロセスの「ライトバージョン」です

require 'rexml/source'

LETTER = '[:alpha:]'
COMBININGCHAR = ''
EXTENDER = ''
NCNAME_STR= "[#{LETTER}_:][-[:alnum:]._:#{COMBININGCHAR}#{EXTENDER}]*"

IDENTITY = /^([!\*\w\-]+)(\s+#{NCNAME_STR})?(\s+["'](.*?)['"])?(\s+['"](.*?)["'])?/u
DOCTYPE_PATTERN = /\s*<!DOCTYPE\s+(.*?)(\[|>)/um

string = <<HERE
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE log4j:configuration SYSTEM "log4j.dtd" >
<log4j:configuration>
</log4j:configuration>
HERE
source = REXML::SourceFactory.create_from(string)
md = source.match( DOCTYPE_PATTERN, true )
identity = md[1]
close = md[2]
identity =~ IDENTITY
name = $1
pub_sys = $2.nil? ? nil : $2.strip
long_name = $4.nil? ? nil : $4.strip
uri = $6.nil? ? nil : $6.strip
args = [ :start_doctype, name, pub_sys, long_name, uri ]
p args  # => [:start_doctype, "log4j", nil, nil, nil]

ご覧のとおり、このスニペットは質問のコードと同じ結果を返します。これに加えて、この動作を変更できるパラメーターがスニペットにないことがわかります。

Nokogiri回避策として、ライブラリを使用することをお勧めします。簡単に見ると、そのようなDoctypeを適切に解析できます。

require 'nokogiri'

string = <<HERE
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE log4j:configuration SYSTEM "log4j.dtd" >
<log4j:configuration>
</log4j:configuration>
HERE

doc = Nokogiri::XML(string)
puts doc.internal_subset.to_s
# => <!DOCTYPE log4j:configuration SYSTEM "log4j.dtd">
于 2011-11-02T18:54:02.090 に答える