How do I remove redundant namespace in nested query when using FOR XML PATH

前端 未结 6 1918
长情又很酷
长情又很酷 2020-11-28 12:54

UPDATE: I\'ve discovered there is a Microsoft Connect item raised for this issue here

When using FOR XML PATH

相关标签:
6条回答
  • 2020-11-28 13:14

    I'm bit confusing about all these explanation while declaring a "xmlns:animals" manually is doing the job : Here an example i wrote to generate Open graph meta data

    DECLARE @l_xml as XML;
    SELECT @l_xml = 
    (
    SELECT 'http://ogp.me/ns# fb: http://ogp.me/ns/fb# scanilike: http://ogp.me/ns/fb/scanilike#' as 'xmlns:og',
        (SELECT
            (SELECT 'og:title' as 'property', title as 'content' for xml raw('meta'), TYPE),
            (SELECT 'og:type' as 'property', OpenGraphWebMetadataTypes.name as 'content' for xml raw('meta'), TYPE),
            (SELECT 'og:image' as 'property', image as 'content' for xml raw('meta'), TYPE),
            (SELECT 'og:url' as 'property', url as 'content' for xml raw('meta'), TYPE),
            (SELECT 'og:description' as 'property', description as 'content' for xml raw('meta'), TYPE),
            (SELECT 'og:site_name' as 'property', siteName as 'content' for xml raw('meta'), TYPE),
            (SELECT 'og:appId' as 'property', appId as 'content' for xml raw('meta'), TYPE)
         FROM OpenGraphWebMetaDatas INNER JOIN OpenGraphWebMetadataTypes ON OpenGraphWebMetaDatas.type = OpenGraphWebMetadataTypes.id WHERE THING_KEY = @p_index 
         for xml path('header'), TYPE),
         (SELECT '' as 'body' for xml path(''), TYPE)
         for xml raw('html'), TYPE
    )
    
    RETURN @l_xml 
    

    returning the expected result

    <html xmlns:og="http://ogp.me/ns# fb: http://ogp.me/ns/fb# scanilike: http://ogp.me/ns/fb/scanilike#">
    <header>
    <meta property="og:title" content="The First object"/>
    <meta property="og:type" content="scanilike:tag"/>
    <meta property="og:image" content="http://www.mygeolive.com/images/facebook/facebook-logo.jpg"/>
    <meta property="og:url" content="http://www.scanilike.com/opengraph?id=1"/>
    <meta property="og:description" content="This is the very first object created using the IOThing &amp; ScanILike software. We keep it in file for history purpose. "/>
    <meta property="og:site_name" content="http://www.scanilike.com"/>
    <meta property="og:appId" content="200270673369521"/>
    </header>
    <body/>
    </html>
    

    hope this will help people are searching the web for similar issue. ;-)

    0 讨论(0)
  • 2020-11-28 13:20

    The problem here is compounded by the fact that you cannot directly declare the namespaces manually when using XML PATH. SQL Server will disallow any attribute names beginning with 'xmlns' and any tag names with colons in them.

    Rather than having to resort to using the relatively unfriendly XML EXPLICIT, I got around the problem by first generating XML with 'cloaked' namespace definitions and references, then doing string replaces as follows ...

    DECLARE @Order TABLE (
      OrderID INT, 
      OrderDate DATETIME)
    
    DECLARE @OrderDetail TABLE (
      OrderID INT, 
      ItemID VARCHAR(1), 
      ItemName VARCHAR(50), 
      Qty INT)
    
    INSERT @Order 
    VALUES 
    (1, '2010-01-01'),
    (2, '2010-01-02')
    
    INSERT @OrderDetail 
    VALUES 
    (1, 'A', 'Drink',  5),
    (1, 'B', 'Cup',    2),
    (2, 'A', 'Drink',  2),
    (2, 'C', 'Straw',  1),
    (2, 'D', 'Napkin', 1)
    
    declare @xml xml
    
    set @xml = (SELECT
      'http://test.com/order' as "@xxmlns..od",  -- 'Cloaked' namespace def
      (SELECT OrderID AS "@OrderID", 
        (SELECT 
          ItemID AS "@od..ItemID", 
          ItemName AS "data()" 
         FROM @OrderDetail 
         WHERE OrderID = o.OrderID 
         FOR XML PATH ('od..Item'), TYPE)
       FROM @Order o
       FOR XML PATH ('od..Order'), TYPE)
      FOR XML PATH('xml'))
    
    set @xml = cast(replace(replace(cast(@xml as nvarchar(max)), 'xxmlns', 'xmlns'),'..',':') as xml)
    
    select @xml
    

    A few things to point out:

    1. I'm using 'xxmlns' as my cloaked version of 'xmlns' and '..' to stand in for ':'. This might not work for you if you're likely to have '..' as part of text values - you can substitute this with something else as long as you pick something that makes a valid XML identifier.

    2. Since we want the xmlns definition at the top level, we cannot use the 'ROOT' option to XML PATH - instead I needed to add an another outer level to the subselect structure to achieve this.

    0 讨论(0)
  • 2020-11-28 13:22

    An alternative solution I've seen is to add the XMLNAMESPACES declaration after building the xml into a temporary variable:

    declare @xml as xml;
    select @xml = (
    select 
        a.c2 as "@species"
        , (select l.c3 as "text()" 
           from t2 l where l.c2 = a.c1 
           for xml path('leg'), type) as "legs"
    from t1 a
    for xml path('animal'))
    
    ;with XmlNamespaces( 'uri:animal' as an)
    select @xml for xml path('') , root('zoo');
    
    0 讨论(0)
  • 2020-11-28 13:32

    After hours of desperation and hundreds of trials & errors, I've come up with the solution below.

    I had the same issue, when I wanted just one xmlns attribute, on the root node only. But I also had a very difficult query with lot's of subqueries and FOR XML EXPLICIT method alone was just too cumbersome. So yes, I wanted the convenience of FOR XML PATH in the subqueries and also to set my own xmlns.

    I kindly borrowed the code of 8kb's answer, because it was so nice. I tweaked it a bit for better understanding. Here is the code:

    DECLARE @Order TABLE (OrderID INT, OrderDate DATETIME)    
    DECLARE @OrderDetail TABLE (OrderID INT, ItemID VARCHAR(1), Name VARCHAR(50), Qty INT)    
    INSERT @Order VALUES (1, '2010-01-01'), (2, '2010-01-02')    
    INSERT @OrderDetail VALUES (1, 'A', 'Drink',  5),
                               (1, 'B', 'Cup',    2),
                               (2, 'A', 'Drink',  2),
                               (2, 'C', 'Straw',  1),
                               (2, 'D', 'Napkin', 1)
    
    -- Your ordinary FOR XML PATH query
    DECLARE @xml XML = (SELECT OrderID AS "@OrderID",
                            (SELECT ItemID AS "@ItemID", 
                                    Name AS "data()" 
                             FROM @OrderDetail 
                             WHERE OrderID = o.OrderID 
                             FOR XML PATH ('Item'), TYPE)
                        FROM @Order o 
                        FOR XML PATH ('Order'), ROOT('dummyTag'), TYPE)
    
    -- Magic happens here!       
    SELECT 1 AS Tag
          ,NULL AS Parent
          ,@xml AS [xml!1!!xmltext]
          ,'http://test.com/order' AS [xml!1!xmlns]
    FOR XML EXPLICIT
    

    Result:

    <xml xmlns="http://test.com/order">
      <Order OrderID="1">
        <Item ItemID="A">Drink</Item>
        <Item ItemID="B">Cup</Item>
      </Order>
      <Order OrderID="2">
        <Item ItemID="A">Drink</Item>
        <Item ItemID="C">Straw</Item>
        <Item ItemID="D">Napkin</Item>
      </Order>
    </xml>
    

    If you selected @xml alone, you would see that it contains root node dummyTag. We don't need it, so we remove it by using directive xmltext in FOR XML EXPLICIT query:

    ,@xml AS [xml!1!!xmltext]
    

    Although the explanation in MSDN sounds more sophisticated, but practically it tells the parser to select the contents of XML root node.

    Not sure how fast the query is, yet currently I am relaxing and drinking Scotch like a gent while peacefully looking at the code...

    0 讨论(0)
  • 2020-11-28 13:32

    It would be really nice if FOR XML PATH actually worked more cleanly. Reworking your original example with @table variables:

    declare @t1 table (c1 int, c2 varchar(50));
    declare @t2 table (c1 int, c2 int, c3 varchar(50));
    insert @t1 values 
        (1, 'Mouse'),
        (2, 'Chicken'),
        (3, 'Snake');
    insert @t2 values
        (1, 1, 'Front Right'),
        (2, 1, 'Front Left'),
        (3, 1, 'Back Right'),
        (4, 1, 'Back Left'),
        (5, 2, 'Right'),
        (6, 2, 'Left');
    
    ;with xmlnamespaces( default 'uri:animal')
    select  a.c2 as "@species",
        (
            select  l.c3 as "text()"
            from    @t2 l
            where   l.c2 = a.c1
            for xml path('leg'), type
        ) as "legs"
    from @t1 a
    for xml path('animal'), root('zoo');
    

    Yields the problem XML with repeated namespace declarations:

    <zoo xmlns="uri:animal">
      <animal species="Mouse">
        <legs>
          <leg xmlns="uri:animal">Front Right</leg>
          <leg xmlns="uri:animal">Front Left</leg>
          <leg xmlns="uri:animal">Back Right</leg>
          <leg xmlns="uri:animal">Back Left</leg>
        </legs>
      </animal>
      <animal species="Chicken">
        <legs>
          <leg xmlns="uri:animal">Right</leg>
          <leg xmlns="uri:animal">Left</leg>
        </legs>
      </animal>
      <animal species="Snake" />
    </zoo>
    

    You can migrate elements between namespaces using XQuery with wildcard namespace matching (that is, *:elementName), as below, but it can be quite cumbersome for complex XML:

    ;with xmlnamespaces( default 'http://tempuri.org/this/namespace/is/meaningless' )
    select (
        select  a.c2 as "@species",
            (
                select  l.c3 as "text()"
                from    @t2 l
                where   l.c2 = a.c1
                for xml path('leg'), type
            ) as "legs"
        from @t1 a
        for xml path('animal'), root('zoo'), type
    ).query('declare default element namespace "uri:animal";
    <zoo>
    { for $a in *:zoo/*:animal return
        <animal>
        {attribute species {$a/@species}}
        { for $l in $a/*:legs return
            <legs>
            { for $m in $l/*:leg return
                <leg>{ $m/text() }</leg>
            }</legs>
        }</animal>
    }</zoo>');
    

    Which yields your desired result:

    <zoo xmlns="uri:animal">
      <animal species="Mouse">
        <legs>
          <leg>Front Right</leg>
          <leg>Front Left</leg>
          <leg>Back Right</leg>
          <leg>Back Left</leg>
        </legs>
      </animal>
      <animal species="Chicken">
        <legs>
          <leg>Right</leg>
          <leg>Left</leg>
        </legs>
      </animal>
      <animal species="Snake" />
    </zoo>
    
    0 讨论(0)
  • 2020-11-28 13:36

    If I have understood correctly, you are referring to the behavior that you might see in a query like this:

    DECLARE @Order TABLE (
      OrderID INT, 
      OrderDate DATETIME)
    
    DECLARE @OrderDetail TABLE (
      OrderID INT, 
      ItemID VARCHAR(1), 
      ItemName VARCHAR(50), 
      Qty INT)
    
    INSERT @Order 
    VALUES 
    (1, '2010-01-01'),
    (2, '2010-01-02')
    
    INSERT @OrderDetail 
    VALUES 
    (1, 'A', 'Drink',  5),
    (1, 'B', 'Cup',    2),
    (2, 'A', 'Drink',  2),
    (2, 'C', 'Straw',  1),
    (2, 'D', 'Napkin', 1)
    
    ;WITH XMLNAMESPACES('http://test.com/order' AS od) 
    SELECT
      OrderID AS "@OrderID",
      (SELECT 
         ItemID AS "@od:ItemID", 
         ItemName AS "data()" 
       FROM @OrderDetail 
       WHERE OrderID = o.OrderID 
       FOR XML PATH ('od.Item'), TYPE)
    FROM @Order o 
    FOR XML PATH ('od.Order'), TYPE, ROOT('xml')
    

    Which gives the following results:

    <xml xmlns:od="http://test.com/order">
      <od.Order OrderID="1">
        <od.Item xmlns:od="http://test.com/order" od:ItemID="A">Drink</od.Item>
        <od.Item xmlns:od="http://test.com/order" od:ItemID="B">Cup</od.Item>
      </od.Order>
      <od.Order OrderID="2">
        <od.Item xmlns:od="http://test.com/order" od:ItemID="A">Drink</od.Item>
        <od.Item xmlns:od="http://test.com/order" od:ItemID="C">Straw</od.Item>
        <od.Item xmlns:od="http://test.com/order" od:ItemID="D">Napkin</od.Item>
      </od.Order>
    </xml>
    

    As you said, the namespace is repeated in the results of the subqueries.

    This behavior is a feature according to a conversation on devnetnewsgroup (website now defunct) although there is the option to vote on changing it.

    My proposed solution is to revert back to FOR XML EXPLICIT:

    SELECT
      1 AS Tag,
      NULL AS Parent,
      'http://test.com/order' AS [xml!1!xmlns:od],
      NULL AS [od:Order!2],
      NULL AS [od:Order!2!OrderID],
      NULL AS [od:Item!3],
      NULL AS [od:Item!3!ItemID]
    UNION ALL
    SELECT 
      2 AS Tag,
      1 AS Parent,
      'http://test.com/order' AS [xml!1!xmlns:od],
      NULL AS [od:Order!2],
      OrderID AS [od:Order!2!OrderID],
      NULL AS [od:Item!3],
      NULL [od:Item!3!ItemID]
    FROM @Order 
    UNION ALL
    SELECT
      3 AS Tag,
      2 AS Parent,
      'http://test.com/order' AS [xml!1!xmlns:od],
      NULL AS [od:Order!2],
      o.OrderID AS [od:Order!2!OrderID],
      d.ItemName AS [od:Item!3],
      d.ItemID AS [od:Item!3!ItemID]
    FROM @Order o INNER JOIN @OrderDetail d ON o.OrderID = d.OrderID
    ORDER BY [od:Order!2!OrderID], [od:Item!3!ItemID]
    FOR XML EXPLICIT
    

    And see these results:

    <xml xmlns:od="http://test.com/order">
      <od:Order OrderID="1">
        <od:Item ItemID="A">Drink</od:Item>
        <od:Item ItemID="B">Cup</od:Item>
      </od:Order>
      <od:Order OrderID="2">
        <od:Item ItemID="A">Drink</od:Item>
        <od:Item ItemID="C">Straw</od:Item>
        <od:Item ItemID="D">Napkin</od:Item>
      </od:Order>
    </xml>
    
    0 讨论(0)
提交回复
热议问题