1
0
Fork 0
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:
kou 2018-04-28 01:36:18 +00:00
parent 75f4f699fd
commit 1d67b9de28
10 changed files with 28 additions and 34 deletions

View file

@ -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

View file

@ -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)

View file

@ -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

View file

@ -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

View file

@ -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>"

View file

@ -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|

View 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

View file

@ -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

View file

@ -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",

View file

@ -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)