XSL проблема с побегом персонажа - PullRequest
8 голосов
/ 14 марта 2009

Я пишу это, потому что я действительно ударил стену и не могу идти вперед. В моей базе данных я экранировал HTML следующим образом: "<p>My name is Freddy and I was".

Я хочу показать его как HTML ИЛИ удалить теги HTML в моем шаблоне XSL. Оба решения будут работать для меня, и я выберу более быстрое решение.

Я прочитал несколько сообщений онлайн, но не могу найти решение. Я также попытался отключить выходной выход без успеха. В основном, кажется, что проблема в том, что где-то в XSL-исполнении движок изменяет это <p> на следующее: <p>.

Преобразует & в &. Если это поможет, вот мой код XSL. Я пробовал несколько комбинаций с тегом вывода и без него сверху.

Любая помощь будет оценена. Заранее спасибо.

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

  <xsl:output method="html" omit-xml-declaration="yes"/>

  <xsl:template match="DocumentElement">
    <div>
      <xsl:attribute name="id">mySlides</xsl:attribute>
      <xsl:apply-templates>
        <xsl:with-param name="templatenumber" select="0"/>
      </xsl:apply-templates>
    </div>

    <div>
      <xsl:attribute name="id">myController</xsl:attribute>
      <xsl:apply-templates>
        <xsl:with-param name="templatenumber" select="1"/>
      </xsl:apply-templates>
    </div>
  </xsl:template>

  <xsl:template match="DocumentElement/QueryResults">
    <xsl:param name="templatenumber">tobereplace</xsl:param>

    <xsl:if test="$templatenumber=0">
      <div>
        <xsl:attribute name="id">myController</xsl:attribute>
        <div>
          <xsl:attribute name="class">article</xsl:attribute>
          <h2>
            <a>
              <xsl:attribute name="class">title</xsl:attribute>
              <xsl:attribute name="title"><xsl:value-of select="Title"/></xsl:attribute>
              <xsl:attribute name="href">/stories/stories-details/articletype/articleview/articleid/<xsl:value-of select="ArticleId"/>/<xsl:value-of select="SEOTitle"/>.aspx</xsl:attribute>
              <xsl:value-of select="Title"/>
            </a>
          </h2>
          <div>
            <xsl:attribute name="style">text-indent: 25px;</xsl:attribute>
            <xsl:attribute name="class">articlesummary</xsl:attribute>
            <xsl:call-template name="removeHtmlTags">
              <xsl:with-param name="html" select="Summary" />
            </xsl:call-template>
          </div>
        </div>
      </div>
    </xsl:if>
    <xsl:if test="$templatenumber=1">
      <div>
        <xsl:attribute name="id">myController</xsl:attribute>
        <span>
          <xsl:attribute name="class">jFlowControl</xsl:attribute>
          aa
        </span>
      </div>
    </xsl:if>
  </xsl:template>

  <xsl:template name="removeHtmlTags">
    <xsl:param name="html"/>
    <xsl:choose>
      <xsl:when test="contains($html, '&lt;')">
        <xsl:value-of select="substring-before($html, '&lt;')"/>
        <!-- Recurse through HTML -->
        <xsl:call-template name="removeHtmlTags">
          <xsl:with-param name="html" select="substring-after($html, '&gt;')"/>
        </xsl:call-template>
      </xsl:when>
      <xsl:otherwise>
        <xsl:value-of select="$html"/>
      </xsl:otherwise>
    </xsl:choose>
  </xsl:template>
</xsl:stylesheet>

Ответы [ 4 ]

18 голосов
/ 15 марта 2009

Исходя из предположения, что у вас есть эта строка HTML,

<p>My name is Freddy &amp; I was

тогда, если вы избежите его и сохраните в базе данных, оно станет this :

&lt;p&gt;My name is Freddy &amp;amp; I was

Следовательно, если вы извлекаете его как XML (без предварительного удаления), результат будет this :

&amp;lt;p&amp;gt;My name is Freddy &amp;amp;amp; I was

и <xsl:value-of select="." disable-output-escaping="yes" /> будут производить:

&lt;p&gt;My name is Freddy &amp;amp; I was

Вы получаете то же самое, что и в своей базе данных, но, конечно, вы видите HTML-теги в выводе. Итак, вам нужен механизм, который выполняет следующие замены строк:

  • "&amp;lt;" с "&lt;" (эффективное изменение &lt; на < в выходном сигнале без экранирования)
  • "&amp;gt;" с "&gt;" (эффективное изменение &gt; на > в выходном сигнале без экранирования)
  • "&amp;quot;" с "&quot;" (эффективное изменение &quot; на " в выходном сигнале без экранирования)
  • "&amp;amp;" с "&amp;" (эффективное изменение &amp; на & в выходном сигнале без экранирования)

Из вашего XSL я вывел следующий тестовый ввод XML:

<DocumentElement>
  <QueryResults>
    <Title>Article 1</Title>
    <ArticleId>1</ArticleId>
    <SEOTitle>Article_1</SEOTitle>
    <Summary>&amp;lt;p&amp;gt;Article 1 summary &amp;amp;amp; description.&amp;lt;/p&amp;gt;</Summary>
  </QueryResults>
  <QueryResults>
    <Title>Article 2</Title>
    <ArticleId>2</ArticleId>
    <SEOTitle>Article_2</SEOTitle>
    <Summary>&amp;lt;p&amp;gt;Article 2 summary &amp;amp;amp; description.&amp;lt;/p&amp;gt;</Summary>
  </QueryResults>
</DocumentElement>

Я изменил предоставленную вами таблицу стилей и реализовал такой механизм замены. Если вы примените к нему следующий шаблон XSLT 1.0:

<xsl:stylesheet
  version="1.0"
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  xmlns:my="my:namespace"
  exclude-result-prefixes="my"
>

  <xsl:output method="html" omit-xml-declaration="yes"/>

  <my:unescape>
    <my:char literal="&lt;" escaped="&amp;lt;" />
    <my:char literal="&gt;" escaped="&amp;gt;" />
    <my:char literal="&quot;" escaped="&amp;quot;" />
    <my:char literal="&amp;" escaped="&amp;amp;" />
  </my:unescape>

  <xsl:template match="DocumentElement">
    <div id="mySlides">
      <xsl:apply-templates mode="slides" />
    </div>
    <div id="myController">
      <xsl:apply-templates mode="controller" />
    </div>
  </xsl:template>

  <xsl:template match="DocumentElement/QueryResults" mode="slides">
    <div class="article">
      <h2>
        <a class="title" title="{Title}" href="{concat('/stories/stories-details/articletype/articleview/articleid/', ArticleId, '/', SEOTitle, '.aspx')}">
          <xsl:value-of select="Title"/>
        </a>
      </h2>
      <div class="articlesummary" style="text-indent: 25px;">
        <xsl:apply-templates select="document('')/*/my:unescape/my:char[1]">
          <xsl:with-param name="html" select="Summary" />
        </xsl:apply-templates>
      </div>
    </div>
  </xsl:template>

  <xsl:template match="DocumentElement/QueryResults" mode="controller">
    <span class="jFlowControl">
      <xsl:text>aa </xsl:text>
      <xsl:value-of select="Title" />
    </span>
  </xsl:template>

  <xsl:template match="my:char">
    <xsl:param name="html" />
    <xsl:variable name="intermediate">
      <xsl:choose>
        <xsl:when test="following-sibling::my:char">
          <xsl:apply-templates select="following-sibling::my:char[1]">
            <xsl:with-param name="html" select="$html" />
          </xsl:apply-templates>
        </xsl:when>
        <xsl:otherwise>
          <xsl:value-of select="$html" disable-output-escaping="yes" />
        </xsl:otherwise>
      </xsl:choose>
    </xsl:variable>
    <xsl:call-template name="unescape">
      <xsl:with-param name="html" select="$intermediate" />
    </xsl:call-template>
  </xsl:template>

  <xsl:template name="unescape">
    <xsl:param name="html" />
    <xsl:choose>
      <xsl:when test="contains($html, @escaped)">
        <xsl:value-of select="substring-before($html, @escaped)" disable-output-escaping="yes"/>
        <xsl:value-of select="@literal" disable-output-escaping="yes" />
        <xsl:call-template name="unescape">
          <xsl:with-param name="html" select="substring-after($html, @escaped)"/>
        </xsl:call-template>
      </xsl:when>
      <xsl:otherwise>
        <xsl:value-of select="$html" disable-output-escaping="yes"/>
      </xsl:otherwise>
    </xsl:choose>
  </xsl:template>

</xsl:stylesheet>

Затем выводится HTML:

<div id="mySlides">
  <div class="article">
    <h2>
      <a class="title" title="Article 1" href="/stories/stories-details/articletype/articleview/articleid/1/Article_1.aspx">Article 1</a>
    </h2>
    <div class="articlesummary" style="text-indent: 25px;">
      <p>Article 1 summary &amp; description.</p>
    </div>
  </div>
  <div class="article">
    <h2>
      <a class="title" title="Article 2" href="/stories/stories-details/articletype/articleview/articleid/2/Article_2.aspx">Article 2</a>
    </h2>
    <div class="articlesummary" style="text-indent: 25px;">
      <p>Article 2 summary &amp; description.</p>
    </div>
  </div>
</div>
<div id="myController">
  <span class="jFlowControl">aa Article 1</span>
  <span class="jFlowControl">aa Article 2</span>
</div>

Примечание

  • использование временного пространства имен и встроенных элементов (<my:unescape>) для создания списка символов для замены
  • использование рекурсии для эмуляции итеративной замены всех задействованных символов во входных данных
  • использование неявного контекста в шаблоне unescape для передачи информации о том, какой символ должен быть заменен в данный момент

Кроме того, примечание:

  • использование шаблонных режимов для получения разных выходных данных для одного и того же ввода (это заменяет ваш параметр templatenumber)
  • в большинстве случаев нет необходимости в <xsl:attribute> элементах. Их можно смело заменять встроенными обозначениями (attributename="{attributevalue}")
  • использование функции concat() для создания URL

Вообще говоря, хранить экранированный HTML в базе данных - плохая идея (в более общем смысле: хранить HTML в базе данных - плохая идея). Вы настроены на то, чтобы получать всевозможные проблемы, и это одна из них. Если вы не можете изменить эту настройку, я надеюсь, что решение поможет вам.

Я не могу гарантировать, что он работает правильно во всех ситуациях, и это может открыть дыры в безопасности (например, XSS), но решение этой проблемы не было частью вопроса. В любом случае считайте себя предупрежденным.

Мне нужен перерыв сейчас. ; -)

6 голосов
/ 14 марта 2009

Вы не должны хранить экранированный HTML в вашей базе данных. Если ваша база данных содержит символ «<», то команда «disable-output-escaping» будет делать то, что вы хотели. </p>

Если вы не можете изменить данные, вам придется удалить данные перед выполнением преобразования.

2 голосов
/ 17 января 2012

Добавьте эту строку в таблицу стилей

<xsl:output method="html" indent="yes" version="4.0"/>
1 голос
/ 01 февраля 2010

Неправильно хранить HTML в базе данных

Что? Как вы должны хранить его тогда? В документе XML, так что вы все равно должны использовать XSLT? Как веб-разработчик, мы всегда использовали базы данных SQL для хранения пользовательских HTML-данных. В этом методе нет ничего плохого, если он правильно очищен для ваших целей.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...