Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

xpath() around 10x slower in JRuby #741

Closed
ghostganz opened this issue Aug 1, 2012 · 28 comments
Closed

xpath() around 10x slower in JRuby #741

ghostganz opened this issue Aug 1, 2012 · 28 comments

Comments

@ghostganz
Copy link

For some reason the xpath() method is a lot slower in nokogiri-java 1.5.5 than the MRI version.

I have code that processes around 1.5 MB of XML, using a lot of nested xpath loops. It takes only 1.7 seconds in MRI but around 150 seconds in JRuby 1.6.7.2 on Java 7. Running with VisualVM shows that almost all time is spent in XmlXpathContext.tryGetNodeSet().

I think this is a regression, because I don't recall this being so slow with older versions of Nokogiri.

@jvshahid
Copy link
Member

jvshahid commented Aug 1, 2012

Can you send an example document and code sample that does xpath queries similar to yours.

@brianjriddle
Copy link

Here is an example repo that exposes the problem

https://github.com/TV4/nokogiri-xpath-issue-741

The repo includes an example xml file and code that exposes the problem.

@ghostganz
Copy link
Author

Using the example code above, I found that this is a regression introduced with nokogiri 1.5.0. It runs much faster with nokogiri 1.4.7

@jvshahid
Copy link
Member

jvshahid commented Aug 2, 2012

thanks, I'll take a look.

@jvshahid
Copy link
Member

jvshahid commented Aug 2, 2012

This isn't really a regression, Nokogiri used FFI prior to 1.5.0. Newer versions of Nokogiri have a pure java implementation which looks to be awfully slower than the C implementation.

@yokolet this looks like a performance issue with Java's JAXP XPath implementation, see this bug report. What do you think ?

@yokolet
Copy link
Member

yokolet commented Aug 2, 2012

hmmm, I agree with Java's XPath implementation is very slow.

I did some research. There might be a couple of ways to improve performance.
For example, http://stackoverflow.com/questions/6340802/java-xpath-apache-jaxp-implementation-performance .

Also, we can cache compiled XPathExpression. Nokogiri's xpath method calls XmlXpathContext.evaluate method, which compiles a given xpath.

The example code repeatedly calls xpath method with the same xpath string. Our implementation compiles the same xpath every time outer each gives a next element. We can avoid this.

@jvshahid
Copy link
Member

jvshahid commented Aug 2, 2012

did you mean XmlXpathContext.compile ? if that's the case then i don't
think compile is the bottle neck. most of the time is spent in evaluate
which uses the compiled expression.

@yokolet
Copy link
Member

yokolet commented Aug 3, 2012

I know the real bottle neck is xpath evaluation, which is on line 143 in XmlXpathContext.java. Do you think we can improve xpath evaluation itself?

I think that's very hard. So, I thought of other options we might improve performance.

Slight possibility is Xerces/JAXP options such as om.sun.org.apache.xml.internal.dtm.DTMManage might help performance.

@jvshahid
Copy link
Member

i looked further into this problem, one of the problems is that we're creating a new XPathContext every time we evaluate the xpath expression which is very expensive. I looked at a couple of libraries and Xalan seems to be the best option compatible with DOM (i.e. doesn't use an internal implementation).

One solution is to use CachedXpathAPI which has the side effect of not exposing setter functions for function and variable resolvers. The other option is to use the internal api (may be subclass CachedXpathAPI). The only drawback is that Nokogiri have to maintain the XPathContext and invalidates it when the document changes, otherwise xpath expressions may return invalid results.

I tested the CachedXpathContext on my machine and ran the script in 23 seconds on JRuby as opposed to 0.7 seconds on MRI. This is better than the current state of the java implementation of Nokogiri which runs the script in 130 seconds on my machine.

what do you guys think ?

@evanworley
Copy link

I've seen this in other platforms/languages, and providing the ability to pre-compile and use cached xpath expressions can greatly speed up processing time. I'm currently working on a project where we are running 10-20 xpath expressions per document, over 1M documents. The penalty for compiling the xpath expression string every time is sever (20M invocations). I'd propose exposing an interface to allow xpath compilation, and then updating the element.xpath method to take either strings or pre-compiled xpath expressions.

@jrochkind
Copy link
Contributor

The much slower performance of jruby nokogiri is a problem for cross-platform use.

Moving from FFI to native java, one would have thought, would make nokogiri work more reliably and higher performance on jruby -- but it's had the opposite effect, making me not want to use nokogiri on jruby due to performance, and making nokogiri even less suitable for a cross-platform solution.

Any hope for any work on this?

@flavorjones
Copy link
Member

Hi Jonathan,

Are you trolling? I honestly can't tell. The FFI port was deprecated a few
years ago. If leaving that behind made Nokogiri unusable, I would have
expected to hear from you sooner on this topic.

The state of the world is, that there's one feature, admittedly a
commonly-used one, that's slower in native Java than in C, and there's a
proposed solution under review.

-mike

On Tue, Jul 9, 2013 at 1:20 PM, Jonathan Rochkind
notifications@github.comwrote:

The much slower performance of jruby nokogiri is a problem for
cross-platform use.

Moving from FFI to native java, one would have thought, would make
nokogiri work more reliably and higher performance on jruby -- but it's had
the opposite effect, making me not want to use nokogiri on jruby due to
performance, and making nokogiri even less suitable for a cross-platform
solution.

Any hope for any work on this?


Reply to this email directly or view it on GitHubhttps://github.com//issues/741#issuecomment-20690360
.

@flavorjones
Copy link
Member

In any case, @jvshahid and @yokolet, can we make this a priority for an upcoming release?

@ghostganz
Copy link
Author

Leaving FFI for native Java did make nokogiri unusable for us, that's why I posted this bug in the first place.

These days we no longer need to parse XML documents, otherwise we'd still be stuck at 1.4.

jvshahid added a commit that referenced this issue Jul 15, 2013
@jvshahid
Copy link
Member

I made some major modifications to the JRuby xpath code. The code on the issue-741 branch uses the internal xpath api and caches the XPathContext so we don't have to create the expensive DTMManager every time we evaluate a xpath expression. There are two drawbacks of this approach:

  1. We're using an internal api
  2. We now have a cached XPathContext. That means unless we have to be very careful to clear the cache everytime the dom is modified, xpath expressions might return nodes that don't exist anymore in the tree or fail to find new nodes added to the dom if we didn't clear the cache.

Currently the entire test suite passes. We had good test coverage on most of the operations that modify the dom, but I'm worried I missed a couple of them. @yokolet can you help eyeball the changes.

By the way, the test time went down from 80 seconds to 6 seconds. It's still slower than MRI (which takes less than a second), but I think it's good enough for the first pass.

Cheers,

@ryanmt
Copy link

ryanmt commented Nov 15, 2013

For future reference, would just using css instead of xpath be faster? How does that leave jruby vs MRI?

@flavorjones
Copy link
Member

On Fri, Nov 15, 2013 at 1:42 PM, Ryan Taylor notifications@github.comwrote:

For future reference, would just using css instead of xpath be faster? How
does that leave jruby vs MRI?

No. Nokogiri compiles CSS queries to XPath before execution, as underlying
XML libraries don't support CSS natively.


Reply to this email directly or view it on GitHubhttps://github.com//issues/741#issuecomment-28592918
.

This was referenced Nov 19, 2013
@knu
Copy link
Member

knu commented Apr 9, 2014

@jvshahid,

test_slow_jruby_xpath is failing too often recently.
Looks like 10 seconds is not enough for Travis to complete the test.
https://travis-ci.org/sparklemotion/nokogiri/jobs/22595529#L178

Was there any regression or is it just because Travis has become slow in performance?

@flavorjones
Copy link
Member

I believe this is Travis running on slow machines. The test suite takes
anywhere from 20 seconds to run (good) to 50 seconds (which causes this
test to fail).
On Apr 9, 2014 5:49 AM, "Akinori MUSHA" notifications@github.com wrote:

@jvshahid https://github.com/jvshahid,

test_slow_jruby_xpath is failing too often recently.
Looks like 10 seconds is not enough for Travis to complete the test.
https://travis-ci.org/sparklemotion/nokogiri/jobs/22595529#L178

Was there any regression or is it just because Travis has become slow in
performance?

Reply to this email directly or view it on GitHubhttps://github.com//issues/741#issuecomment-39946359
.

@knu
Copy link
Member

knu commented Apr 9, 2014

OK, let's see if extending the time limit to 20 on Travis CI works... (b094730)

@jvshahid
Copy link
Member

jvshahid commented Apr 9, 2014

We should probably disable the test. I'm not really happy with the current way of measuring performance, it's very brittle and will break on slower machines. Is there a better way to do this ?

@flavorjones
Copy link
Member

@zenspider wrote minitest/benchmark to do something similar to this,
where you can assert {constant,linear,exponential} performance
characteristics of an algorithm.

http://www.zenspider.com/projects/minitest.html#benchmarks

http://docs.seattlerb.org/minitest/Minitest/Benchmark.html

Would that be appropriate?

On Wed, Apr 9, 2014 at 11:57 AM, John Shahid notifications@github.comwrote:

We should probably disable the test. I'm not really happy with the current
way of measuring performance, it's very brittle and will break on slower
machines. Is there a better way to do this ?

Reply to this email directly or view it on GitHubhttps://github.com//issues/741#issuecomment-39981422
.

@jvshahid
Copy link
Member

jvshahid commented Apr 9, 2014

I actually prefer this over timing the operation. The test will have to modified slightly to generate a xml document that's proportional to n. I'll try to get this done over the weekend and go through the backlog of the JRuby stories.

@knu
Copy link
Member

knu commented Apr 10, 2014

Sounds great. Obviously, an absolute time limit wouldn't be the way to go.

I tweaked the value specially for Travis CI for the moment, so unless this is a regression we can move forward to the upcoming release.

@flavorjones
Copy link
Member

@jvshahid this test is failing in the concourse pipelines, because of noisy neighbor problems when all the tests are running at the same time.

Is there a real worry about regression here? If not, maybe we just delete it rather than make it more complicated? I'm sensitive to the amount of work we'd need to put in to make this not-brittle.

@jvshahid
Copy link
Member

I'm not happy with that test either, but I don't know if there's a better way to test performance regression. I'm okay with deleting it. That code doesn't change frequently to justify having the brittle tests.

@flavorjones
Copy link
Member

Roger, gonna skip it for all platforms.

@flavorjones
Copy link
Member

See 2d2893a

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

9 participants