在Umbraco XSLT宏中优雅地处理缺少的XML提要

时间:2012-01-30 17:53:05

标签: xml xslt umbraco

我正在尝试使用Umbraco中的XSLT宏读取XML Feed,并让它显示格式良好的内容。当Feed可用时,我的宏工作正常,但如果Feed返回404,我无法让XSLT正常处理它。

我使用umbraco.library获取XML:GetXmlDocumentByUrl() 我发现它正在创建一个解析错误,有时它只会崩溃网站而不是返回我指定的错误文本。

我还尝试在文档()测试中包装GetXmlDocumentByUrl()以查看是否可以使用它来更好地处理错误。我发现虽然这会阻止站点崩溃,并且如果XML源存在,但它仍然有效,它仍然会创建一个解析错误而不是显示我的错误文本。

我很感激您的任何帮助或建议,我的代码如下:

<xsl:variable name="feed" select="'http://url.to.feed'"/>

<xsl:template match="/">
  <xsl:value-of select="document($feed)"/>
  <!-- start writing XSLT -->
  <xsl:choose>
    <xsl:when test="string-length($feed) > 0 and $feed != ''">
      <xsl:choose>
        <xsl:when test="document($feed)">
           File found
          <xsl:variable name="feedContent" select="umbraco.library:GetXmlDocumentByUrl($feed, $cacheRate)"/>
          <xsl:choose>
            <xsl:when test="count($feedContent/error) &gt; 0">
            <!--<xsl:when test="$feedContent != 'error'">-->
              <p class="feedList">
                <strong>This dynamic content is currently not available</strong><br />
                The content could not be loaded. Please verify that you are on the correct page and that you have an
                active internet connection.
              </p>
            </xsl:when>
            <xsl:otherwise>
              <xsl:call-template name="renderFeed">
                <xsl:with-param name="feedContent" select="$feedContent"/>
              </xsl:call-template>
            </xsl:otherwise>
          </xsl:choose>
        </xsl:when>
        <xsl:otherwise>
            Can't find the file...
        </xsl:otherwise>
      </xsl:choose>
    </xsl:when>
    <xsl:otherwise>
      <p class="feedList">
        <strong>No content exists for this page</strong><br />
        Please view another page.
      </p>
    </xsl:otherwise>
  </xsl:choose>
</xsl:template>

更新: 我试着浏览我的代码以简化以下问题,这应该使用GetXmlDocumentByUrl的非缓存实现,这样我就可以确保我没有问题,并且还可以直接输出值确保这不是我选择的陈述:

 <xsl:template match="/">
  <!-- start writing XSLT -->
  <xsl:choose>
    <xsl:when test="string-length($feed) > 0 and $feed != ''">
      <xsl:variable name="vDoc" select="umbraco.library:GetXmlDocumentByUrl($feed)"/>
      <xsl:value-of select="$vDoc"/>
      <xsl:choose>
        <xsl:when test="$vDoc">
           File found
        </xsl:when>
        <xsl:otherwise>
            Can't find the file...
        </xsl:otherwise>
      </xsl:choose>
    </xsl:when>
    <xsl:otherwise>
      <p class="feedList">
        <strong>No content exists for this page</strong><br />
        Please view another page.
      </p>
    </xsl:otherwise>
  </xsl:choose>
</xsl:template>

有一个明确的404页面,它返回一个字符串“System.Net.WebException:远程服务器返回错误:(404)Not Found。在System.Net.HttpWebRequest.GetResponse()at umbraco.library.GetXmlDocumentByUrl( String Url)“然而,由于我实际上有超时的问题,我用fiddler进行了双重检查,看起来页面实际上是返回200,但不是XML文档,我应该在我的renderFeed中提到模板如下,所以我仍然希望它显示其他内容而不是超时。

 <xsl:template name="renderFeed">
  <xsl:param name="feedContent" />
  <xsl:choose>
    <xsl:when test="count($feedContent//item) &gt; 0">
        //Render Feed content
    </xsl:when>
    <xsl:otherwise>
     <p class="feedList">
        <strong>No content exists for this page</strong><br />
        Please view another page.
      </p>
   </xsl:otherwise>
  </xsl:choose>
</xsl:template>

我从一个例子中得到了测试时间,我有更好的方法来测试吗?

2 个答案:

答案 0 :(得分:0)

以下是W3C XSLT 1.0规范针对document()函数无法检索或解析文档的情况所说的内容:

  

如果检索资源时出错,则XSLT   处理器可能发出错误信号;如果它没有发出错误信号,那就必须   通过返回空节点集来恢复

这个使用特定XSLT处理器的机会不会引发异常,但会静默返回一个空的节点集。

试试这个 - 你可能很幸运

<xsl:stylesheet version="1.0"
 xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

 <xsl:variable name="feed" select="'http://Some.Url.com'"/>

    <xsl:template match="/">
        <xsl:variable name="vDoc" select="document($feed)"/>

        <xsl:choose>
            <xsl:when test="$vDoc">
              <!--Normal processing here  -->
            </xsl:when>
            <xsl:otherwise>
                <p class="feedList">
                    <strong>No content exists for this page</strong>
                    <br />         Please view another page.       
                </p>
            </xsl:otherwise>
        </xsl:choose>
    </xsl:template>
</xsl:stylesheet>

当我使用Saxon对任何XML文档(未使用)进行此转换时,大约1分钟后我得到

<p class="feedList">
    <strong>No content exists for this page</strong>
    <br/>         Please view another page.       
</p>

答案 1 :(得分:0)

我能看到的唯一问题是,XmlDocument加载了ContentType text/html而不是text/xml。我写了一个简单的函数,基于Umbraco中的原始函数,允许你改变WebRequest超时加号,它检查ContentType

public static XPathNodeIterator GetXmlDocumentByUrl(string Url, int requestTimeout = 100000)
{
    XmlDocument xmlDoc = new XmlDocument();
    WebRequest request = WebRequest.Create(Url);

    try
    {
        // Set the Request Timeout
        request.Timeout = requestTimeout;
        using (WebResponse response = request.GetResponse())
        {
            if (response.ContentType.Contains("text/xml"))
            {
                using (Stream responseStream = response.GetResponseStream())
                {
                    XmlTextReader reader = new XmlTextReader(responseStream);
                    xmlDoc.Load(reader);
                }
            }
            else
                xmlDoc.LoadXml(string.Format("<error url=\"{0}\">Failed to load an ContentType of XML</error>",
                                             HttpContext.Current.Server.HtmlEncode(Url)));
        }
    }
    catch (WebException err)
    {
        xmlDoc.LoadXml(string.Format("<error url=\"{0}\">{1}</error>",
                                     HttpContext.Current.Server.HtmlEncode(Url), err));
    }
    catch (Exception err)
    {
        xmlDoc.LoadXml(string.Format("<error url=\"{0}\">{1}</error>",
                                     HttpContext.Current.Server.HtmlEncode(Url), err));
    }

    XPathNavigator xp = xmlDoc.CreateNavigator();
    return xp.Select("/");
}