mirror of
https://github.com/ruby/ruby.git
synced 2022-11-09 12:17:21 -05:00
rexml: disable XPath 1.0 compatible "#{ELEMENT_NAME}" processing by default
It breaks backward compatibility than I thought. So it's disabled by default. It means that REXML's XPath processor isn't compatible with XPath 1.0. But it will be acceptable for users. We can enable it by specifying "strict: true" to REXML::XPathParser.new explicitly. * lib/rexml/xpath.rb, lib/rexml/xpath_parser.rb: Accept "strict: true" option. * test/rexml/test_contrib.rb, test/rexml/xpath/test_base.rb: Use not XPath 1.0 compatible behavior. * test/rexml/test_jaxen.rb: Use XPath 1.0 compatible behavior. * test/rss/test_1.0.rb, test/rss/test_dublincore.rb, spec/ruby/library/rexml/element/namespace_spec.rb, spec/ruby/library/rexml/element/namespaces_spec.rb, spec/ruby/library/rexml/element/prefixes_spec.rb: Enable again. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@63278 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
75f4f699fd
commit
1d67b9de28
10 changed files with 28 additions and 34 deletions
|
@ -28,10 +28,10 @@ module REXML
|
||||||
# XPath.first( doc, "//b"} )
|
# XPath.first( doc, "//b"} )
|
||||||
# XPath.first( node, "a/x:b", { "x"=>"http://doofus" } )
|
# XPath.first( node, "a/x:b", { "x"=>"http://doofus" } )
|
||||||
# XPath.first( node, '/book/publisher/text()=$publisher', {}, {"publisher"=>"O'Reilly"})
|
# XPath.first( node, '/book/publisher/text()=$publisher', {}, {"publisher"=>"O'Reilly"})
|
||||||
def XPath::first element, path=nil, namespaces=nil, variables={}
|
def XPath::first(element, path=nil, namespaces=nil, variables={}, options={})
|
||||||
raise "The namespaces argument, if supplied, must be a hash object." unless namespaces.nil? or namespaces.kind_of?(Hash)
|
raise "The namespaces argument, if supplied, must be a hash object." unless namespaces.nil? or namespaces.kind_of?(Hash)
|
||||||
raise "The variables argument, if supplied, must be a hash object." unless variables.kind_of?(Hash)
|
raise "The variables argument, if supplied, must be a hash object." unless variables.kind_of?(Hash)
|
||||||
parser = XPathParser.new
|
parser = XPathParser.new(**options)
|
||||||
parser.namespaces = namespaces
|
parser.namespaces = namespaces
|
||||||
parser.variables = variables
|
parser.variables = variables
|
||||||
path = "*" unless path
|
path = "*" unless path
|
||||||
|
@ -57,10 +57,10 @@ module REXML
|
||||||
# XPath.each( node, 'ancestor::x' ) { |el| ... }
|
# XPath.each( node, 'ancestor::x' ) { |el| ... }
|
||||||
# XPath.each( node, '/book/publisher/text()=$publisher', {}, {"publisher"=>"O'Reilly"}) \
|
# XPath.each( node, '/book/publisher/text()=$publisher', {}, {"publisher"=>"O'Reilly"}) \
|
||||||
# {|el| ... }
|
# {|el| ... }
|
||||||
def XPath::each element, path=nil, namespaces=nil, variables={}, &block
|
def XPath::each(element, path=nil, namespaces=nil, variables={}, options={}, &block)
|
||||||
raise "The namespaces argument, if supplied, must be a hash object." unless namespaces.nil? or namespaces.kind_of?(Hash)
|
raise "The namespaces argument, if supplied, must be a hash object." unless namespaces.nil? or namespaces.kind_of?(Hash)
|
||||||
raise "The variables argument, if supplied, must be a hash object." unless variables.kind_of?(Hash)
|
raise "The variables argument, if supplied, must be a hash object." unless variables.kind_of?(Hash)
|
||||||
parser = XPathParser.new
|
parser = XPathParser.new(**options)
|
||||||
parser.namespaces = namespaces
|
parser.namespaces = namespaces
|
||||||
parser.variables = variables
|
parser.variables = variables
|
||||||
path = "*" unless path
|
path = "*" unless path
|
||||||
|
@ -69,8 +69,8 @@ module REXML
|
||||||
end
|
end
|
||||||
|
|
||||||
# Returns an array of nodes matching a given XPath.
|
# Returns an array of nodes matching a given XPath.
|
||||||
def XPath::match element, path=nil, namespaces=nil, variables={}
|
def XPath::match(element, path=nil, namespaces=nil, variables={}, options={})
|
||||||
parser = XPathParser.new
|
parser = XPathParser.new(**options)
|
||||||
parser.namespaces = namespaces
|
parser.namespaces = namespaces
|
||||||
parser.variables = variables
|
parser.variables = variables
|
||||||
path = "*" unless path
|
path = "*" unless path
|
||||||
|
|
|
@ -47,11 +47,12 @@ module REXML
|
||||||
include XMLTokens
|
include XMLTokens
|
||||||
LITERAL = /^'([^']*)'|^"([^"]*)"/u
|
LITERAL = /^'([^']*)'|^"([^"]*)"/u
|
||||||
|
|
||||||
def initialize( )
|
def initialize(strict: false)
|
||||||
@parser = REXML::Parsers::XPathParser.new
|
@parser = REXML::Parsers::XPathParser.new
|
||||||
@namespaces = nil
|
@namespaces = nil
|
||||||
@variables = {}
|
@variables = {}
|
||||||
@nest = 0
|
@nest = 0
|
||||||
|
@strict = strict
|
||||||
end
|
end
|
||||||
|
|
||||||
def namespaces=( namespaces={} )
|
def namespaces=( namespaces={} )
|
||||||
|
@ -139,7 +140,9 @@ module REXML
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
def strict?
|
||||||
|
@strict
|
||||||
|
end
|
||||||
|
|
||||||
# Returns a String namespace for a node, given a prefix
|
# Returns a String namespace for a node, given a prefix
|
||||||
# The rules are:
|
# The rules are:
|
||||||
|
@ -474,7 +477,13 @@ module REXML
|
||||||
if prefix.nil?
|
if prefix.nil?
|
||||||
raw_node.name == name
|
raw_node.name == name
|
||||||
elsif prefix.empty?
|
elsif prefix.empty?
|
||||||
raw_node.name == name and raw_node.namespace == ""
|
if strict?
|
||||||
|
raw_node.name == name and raw_node.namespace == ""
|
||||||
|
else
|
||||||
|
# FIXME: This DOUBLES the time XPath searches take
|
||||||
|
ns = get_namespace(raw_node, prefix)
|
||||||
|
raw_node.name == name and raw_node.namespace == ns
|
||||||
|
end
|
||||||
else
|
else
|
||||||
# FIXME: This DOUBLES the time XPath searches take
|
# FIXME: This DOUBLES the time XPath searches take
|
||||||
ns = get_namespace(raw_node, prefix)
|
ns = get_namespace(raw_node, prefix)
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
require 'rexml/document'
|
require 'rexml/document'
|
||||||
require_relative '../../../spec_helper'
|
require_relative '../../../spec_helper'
|
||||||
|
|
||||||
=begin
|
|
||||||
# FIXME
|
|
||||||
describe "REXML::Element#namespace" do
|
describe "REXML::Element#namespace" do
|
||||||
before :each do
|
before :each do
|
||||||
@doc = REXML::Document.new("<a xmlns='1' xmlns:y='2'><b/><c xmlns:z='3'/></a>")
|
@doc = REXML::Document.new("<a xmlns='1' xmlns:y='2'><b/><c xmlns:z='3'/></a>")
|
||||||
|
@ -27,4 +25,3 @@ describe "REXML::Element#namespace" do
|
||||||
@elem.namespace("z").should be_nil
|
@elem.namespace("z").should be_nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
=end
|
|
||||||
|
|
|
@ -7,7 +7,6 @@ describe "REXML::Element#namespaces" do
|
||||||
@elem = doc.elements["//c"]
|
@elem = doc.elements["//c"]
|
||||||
end
|
end
|
||||||
|
|
||||||
=begin
|
|
||||||
it "returns a hash of the namespaces" do
|
it "returns a hash of the namespaces" do
|
||||||
ns = {"y"=>"2", "z"=>"3", "xmlns"=>"1"}
|
ns = {"y"=>"2", "z"=>"3", "xmlns"=>"1"}
|
||||||
@elem.namespaces.keys.sort.should == ns.keys.sort
|
@elem.namespaces.keys.sort.should == ns.keys.sort
|
||||||
|
@ -29,5 +28,4 @@ describe "REXML::Element#namespaces" do
|
||||||
values = ["2", "3", "1"]
|
values = ["2", "3", "1"]
|
||||||
@elem.namespaces.values.sort.should == values.sort
|
@elem.namespaces.values.sort.should == values.sort
|
||||||
end
|
end
|
||||||
=end
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -7,7 +7,6 @@ describe "REXML::Element#prefixes" do
|
||||||
@elem = doc.elements["//c"]
|
@elem = doc.elements["//c"]
|
||||||
end
|
end
|
||||||
|
|
||||||
=begin
|
|
||||||
it "returns an array of the prefixes of the namespaces" do
|
it "returns an array of the prefixes of the namespaces" do
|
||||||
@elem.prefixes.should == ["y", "z"]
|
@elem.prefixes.should == ["y", "z"]
|
||||||
end
|
end
|
||||||
|
@ -15,7 +14,6 @@ describe "REXML::Element#prefixes" do
|
||||||
it "does not include the default namespace" do
|
it "does not include the default namespace" do
|
||||||
@elem.prefixes.include?("xmlns").should == false
|
@elem.prefixes.include?("xmlns").should == false
|
||||||
end
|
end
|
||||||
=end
|
|
||||||
|
|
||||||
it "returns an empty array if no namespace was defined" do
|
it "returns an empty array if no namespace was defined" do
|
||||||
doc = REXML::Document.new "<root><something/></root>"
|
doc = REXML::Document.new "<root><something/></root>"
|
||||||
|
|
|
@ -451,7 +451,7 @@ EOL
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_external_entity
|
def test_external_entity
|
||||||
xp = '//*:channel/*:title'
|
xp = '//channel/title'
|
||||||
|
|
||||||
%w{working.rss broken.rss}.each do |path|
|
%w{working.rss broken.rss}.each do |path|
|
||||||
File.open(File.join(fixture_path(path))) do |file|
|
File.open(File.join(fixture_path(path))) do |file|
|
||||||
|
|
|
@ -84,7 +84,7 @@ module REXMLTests
|
||||||
def process_value_of(context, variables, namespaces, value_of)
|
def process_value_of(context, variables, namespaces, value_of)
|
||||||
expected = value_of.text
|
expected = value_of.text
|
||||||
xpath = value_of.attributes["select"]
|
xpath = value_of.attributes["select"]
|
||||||
matched = XPath.match(context, xpath, namespaces, variables)
|
matched = XPath.match(context, xpath, namespaces, variables, strict: true)
|
||||||
|
|
||||||
message = user_message(context, xpath, matched)
|
message = user_message(context, xpath, matched)
|
||||||
assert_equal(expected || "",
|
assert_equal(expected || "",
|
||||||
|
@ -95,7 +95,7 @@ module REXMLTests
|
||||||
# processes a tests/document/context/test node ( where @exception is false or doesn't exist )
|
# processes a tests/document/context/test node ( where @exception is false or doesn't exist )
|
||||||
def process_nominal_test(context, variables, namespaces, test)
|
def process_nominal_test(context, variables, namespaces, test)
|
||||||
xpath = test.attributes["select"]
|
xpath = test.attributes["select"]
|
||||||
matched = XPath.match(context, xpath, namespaces, variables)
|
matched = XPath.match(context, xpath, namespaces, variables, strict: true)
|
||||||
# might be a test with no count attribute, but nested valueOf elements
|
# might be a test with no count attribute, but nested valueOf elements
|
||||||
expected = test.attributes["count"]
|
expected = test.attributes["count"]
|
||||||
if expected
|
if expected
|
||||||
|
@ -113,7 +113,7 @@ module REXMLTests
|
||||||
def process_exceptional_test(context, variables, namespaces, test)
|
def process_exceptional_test(context, variables, namespaces, test)
|
||||||
xpath = test.attributes["select"]
|
xpath = test.attributes["select"]
|
||||||
assert_raise(REXML::ParseException) do
|
assert_raise(REXML::ParseException) do
|
||||||
XPath.match(context, xpath, namespaces, variables)
|
XPath.match(context, xpath, namespaces, variables, strict: true)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -880,10 +880,12 @@ module REXMLTests
|
||||||
<tada xmlns=''>xb</tada>
|
<tada xmlns=''>xb</tada>
|
||||||
</tag1>
|
</tag1>
|
||||||
XML
|
XML
|
||||||
x = d.root
|
actual = []
|
||||||
num = 0
|
d.root.each_element('tada') do |element|
|
||||||
x.each_element('tada') { num += 1 }
|
actual << element.to_s
|
||||||
assert_equal(1, num)
|
end
|
||||||
|
assert_equal(["<tada>xa</tada>", "<tada xmlns=''>xb</tada>"],
|
||||||
|
actual)
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_ticket_39
|
def test_ticket_39
|
||||||
|
|
|
@ -54,8 +54,6 @@ module RSS
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_channel
|
def test_channel
|
||||||
skip # FIXME
|
|
||||||
|
|
||||||
about = "http://hoge.com"
|
about = "http://hoge.com"
|
||||||
resource = "http://hoge.com/hoge.png"
|
resource = "http://hoge.com/hoge.png"
|
||||||
|
|
||||||
|
@ -207,8 +205,6 @@ EOR
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_image
|
def test_image
|
||||||
skip # FIXME
|
|
||||||
|
|
||||||
about = "http://hoge.com"
|
about = "http://hoge.com"
|
||||||
h = {
|
h = {
|
||||||
'title' => "fugafuga",
|
'title' => "fugafuga",
|
||||||
|
@ -234,8 +230,6 @@ EOR
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_item
|
def test_item
|
||||||
skip # FIXME
|
|
||||||
|
|
||||||
about = "http://hoge.com"
|
about = "http://hoge.com"
|
||||||
h = {
|
h = {
|
||||||
'title' => "fugafuga",
|
'title' => "fugafuga",
|
||||||
|
@ -261,8 +255,6 @@ EOR
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_textinput
|
def test_textinput
|
||||||
skip # FIXME
|
|
||||||
|
|
||||||
about = "http://hoge.com"
|
about = "http://hoge.com"
|
||||||
h = {
|
h = {
|
||||||
'title' => "fugafuga",
|
'title' => "fugafuga",
|
||||||
|
|
|
@ -103,8 +103,6 @@ EOR
|
||||||
end
|
end
|
||||||
|
|
||||||
def test_to_s
|
def test_to_s
|
||||||
skip # FIXME
|
|
||||||
|
|
||||||
assert_dc_to_s(@rss10_source, @rss10_parents, false)
|
assert_dc_to_s(@rss10_source, @rss10_parents, false)
|
||||||
assert_dc_to_s(@rss10_source, @rss10_parents, true)
|
assert_dc_to_s(@rss10_source, @rss10_parents, true)
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue