I want to export my Asics running plan to iCal and since Asics do not offer this service, I decided to build a little scraper for my own personal use. What I want to do is to take all the scheduled runs from my plan and generate an iCal feed based on that. I am using C# and Html Agility Pack.
What I want to do is iterate through all my scheduled runs (they are div nodes). Then next I want to select a few different nodes with my run nodes. My code looks like this:
foreach (var run in doc.DocumentNode.SelectSingleNode("//div[@id='scheduleTable']").SelectNodes("//div[@class='pTdBox']"))
{
number++;
string date = run.SelectSingleNode("//div[@class='date']").InnerText;
string type = run.SelectSingleNode("//span[@class='menu']").InnerHtml;
string distance = run.SelectSingleNode("//span[@class='distance']").InnerHtml;
string description = run.SelectSingleNode("//div[@class='description']").In开发者_运维知识库nerHtml;
ViewData["result"] += "Dato: " + date + "<br />";
ViewData["result"] += "Tyep: " + type + "<br />";
ViewData["result"] += "Distance: " + distance + "<br />";
ViewData["result"] += "Description: " + description + "<br />";
ViewData["result"] += run.InnerHtml.Replace("<", "<").Replace(">", ">") + "<br />" + "<br />" + "<br />";
}
My problem is that run.SelectSingleNode("//div[@class='date']").InnerText
does not select the node with the given XPath within the given run node. It selects the first node that matches the XPath in the entire document.
How can I select the single node with the given XPath within the current node?
Thank you.
Update
I tried updating my XPath string to this:
string date = run.SelectSingleNode(".div[@class='date']").InnerText;
This should select the <div class="date"></div>
element within the current node, right? Well, I tried this but got this error:
Expression must evaluate to a node-set. Description: An unhandled exception occurred during the execution of the current web request. Please review the stack trace for more information about the error and where it originated in the code.
Exception Details: System.Xml.XPath.XPathException: Expression must evaluate to a node-set.
Any suggestions?
A few things that will help you when working with HtmlAgilityPack and XPath expressions.
If run
is an HtmlNode
, then:
run.SelectNodes("//div[@class='date']")
Will will behave exactly likedoc.DocumentNode.SelectNodes("//div[@class='date']")
run.SelectNodes("./div[@class='date']")
Will give you all the<div>
nodes that are children ofrun
node. It won't search deeper, only at the very next depth level.run.SelectNodes(".//div[@class='date']")
Will return all the<div>
nodes with that class attribute, but not only next to therun
node, but also will search in depth (every possible descendant of it)
You will have to choose between 2. or 3., depending on which one satisfy your needs :)
In XPATH, //
means all children and grand children below the current node. So you need to come up with a more restrictive XPATH expression. If you provide the real HTML, and what you're looking for exactly, we can help you dig further.
About the error you have:
.div[@class='date']
is invalid because .
is sticked to div
. You could use div[@class='date']
, or ./div[@class='date']
which I believe are equivalent. This is because .
is an XPATH axe, which is an alias for self
and means "the current node".
精彩评论