如何编写 PL 以返回通过服务总线中的 DB 适配器自动生成强类型 XSD 的多行? (甲骨文 PL/SQL)

Posted

技术标签:

【中文标题】如何编写 PL 以返回通过服务总线中的 DB 适配器自动生成强类型 XSD 的多行? (甲骨文 PL/SQL)【英文标题】:How to write PL to return multiple rows which auto generates strong typed XSD via DB Adapter in Service Bus? (Oracle PL/SQL) 【发布时间】:2015-09-17 12:43:05 【问题描述】:

我正在通过 Oracle Service Bus 中的 DB 适配器调用一个过程。我可以使用 SYS_REFCURSOR 作为 OUT 参数从 PL 中选择多行。

/*Package Specification*/

create or replace PACKAGE ACC_HIER AS 

  PROCEDURE f_Hier(
      IN_T_CODE IN HIER.T_CODE%TYPE,
      H_DETAILS OUT SYS_REFCURSOR);

END ACC_HIER;


/*Package Body*/

create or replace PACKAGE BODY ACC_HIER AS


  PROCEDURE f_Hier(
      IN_T_CODE IN HIER.T_CODE%TYPE,
      H_DETAILS OUT SYS_REFCURSOR) AS

  BEGIN

    OPEN H_DETAILS FOR
         SELECT T_LEVEL,T_CODE,P_ID,P_NAME,LAST_UPDATE_DATE
         FROM HIER
         WHERE T_CODE = IN_T_CODE
         ORDER BY T_LEVEL;
    EXCEPTION 
            WHEN OTHERS 
            THEN dbms_output.put_line('No Rows Found');
  END f_Hier;

END ACC_HIER;

当我通过 DB 适配器导入包时,它会自动生成弱 xsd:

<schema targetNamespace="http://xmlns.test.com/pcbpel/adapter/db/sp/DB_AccHierarchy_Pkg" xmlns="http://www.w3.org/2001/XMLSchema" xmlns:db="http://xmlns.test.com/pcbpel/adapter/db/sp/DB_AccHierarchy_Pkg" elementFormDefault="qualified">
   <element name="InputParameters">
      <complexType>
         <sequence>
            <element name="IN_T_CODE" type="string" db:index="1" db:type="VARCHAR2" minOccurs="0" nillable="true"/>
         </sequence>
      </complexType>
   </element>
   <element name="OutputParameters">
      <complexType>
         <sequence>
            <element name="H_DETAILS" type="db:RowSet" db:index="2" db:type="RowSet" minOccurs="0" nillable="true"/>
         </sequence>
      </complexType>
   </element>
   <complexType name="RowSet">
      <sequence>
         <element name="Row" minOccurs="0" maxOccurs="unbounded">
            <complexType>
               <sequence>
                  <element name="Column" maxOccurs="unbounded" nillable="true">
                     <complexType>
                        <simpleContent>
                           <extension base="string">
                              <attribute name="name" type="string" use="required"/>
                              <attribute name="sqltype" type="string" use="required"/>
                           </extension>
                        </simpleContent>
                     </complexType>
                  </element>
               </sequence>
            </complexType>
         </element>
      </sequence>
   </complexType>
</schema>

属性是通用的,当我们执行 DB Adapter 时,输出如下:

<Row>
<Column name="T_LEVEL" sqltype="NUMBER">1</Column>
<Column name="T_CODE" sqltype="VARCHAR2">XYZ</Column>
<Column name="P_ID" sqltype="NUMBER">13214</Column>
<Column name="P_NAME" sqltype="VARCHAR2">XYZ_1 Limited</Column>
<Column name="LAST_UPDATE_DATE" sqltype="TIMESTAMP">2015-07-01T09:21:27.901107+05:30</Column>
</Row>
<Row>
<Column name="T_LEVEL" sqltype="NUMBER">2</Column>
<Column name="T_CODE" sqltype="VARCHAR2">XYZ</Column>
<Column name="P_ID" sqltype="NUMBER">156219</Column>
<Column name="P_NAME" sqltype="VARCHAR2">XYZ_2 Limited</Column>
<Column name="LAST_UPDATE_DATE" sqltype="TIMESTAMP">2015-07-02T09:21:27.901107+05:30</Column>
</Row>

要求:我希望生成的XSD类似于下面的sn-p:

  <complexType>
     <element name="T_LEVEL" type="decimal" nillable="true"/>
     <element name="T_CODE" type="string" nillable="true"/>
     <element name="P_ID" type="decimal" nillable="true"/>
     <element name="P_NAME" type="string" nillable="true"/>
     <element name="LAST_UPDATE_DATE" type="dateTime" nillable="true"/>
  </complexType>

和输出类似:

<Row>
<T_LEVEL>1</T_LEVEL>
<T_CODE>XYZ</T_CODE>
<P_ID>13214</P_ID>
<P_NAME>XYZ_1 Limited</P_NAME>
<LAST_UPDATE_DATE>2015-07-01T09:21:27.901107+05:30</LAST_UPDATE_DATE>
</Row>
<Row>
<T_LEVEL>2</T_LEVEL>
<T_CODE>XYZ</T_CODE>
<P_ID>156219</P_ID>
<P_NAME>XYZ_2 Limited</P_NAME>
<LAST_UPDATE_DATE>2015-07-02T09:21:27.901107+05:30</LAST_UPDATE_DATE>
</Row>

数据库适配器有什么方法可以自动生成所需格式的 XSD? 程序中的哪些变化可以让我获得所需的结果?任何替代解决方案?

【问题讨论】:

【参考方案1】:

Nameet,我认为期望 DB 适配器生成所需格式的 XSD 可能不是正确的方法。

我不知道您是否可以将结果转换为所需的格式,但如果可以,我认为这会有所帮助:

FinalSchema.xsd

<?xml version="1.0" encoding="UTF-8" ?>
<xsd:schema xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns="http://www.example.org"
            targetNamespace="http://www.example.org" elementFormDefault="qualified">
  <xsd:element name="DBRowList" type="DBRowListType"/>

  <xsd:complexType name="DBRowListType">
    <xsd:sequence>
      <xsd:element name="dbRow" type="DBRowType" nillable="true" maxOccurs="unbounded"/>
    </xsd:sequence>
  </xsd:complexType>
  <xsd:complexType name="DBRowType">
    <xsd:sequence>
      <xsd:element name="T_LEVEL" type="xsd:decimal" nillable="true"/>
      <xsd:element name="T_CODE" type="xsd:string" nillable="true"/>
      <xsd:element name="P_ID" type="xsd:decimal" nillable="true"/>
      <xsd:element name="P_NAME" type="xsd:string" nillable="true"/>
      <xsd:element name="LAST_UPDATE_DATE" type="xsd:dateTime" nillable="true"/>
    </xsd:sequence>
  </xsd:complexType>
</xsd:schema>

Transform.xsl

<?xml version="1.0" encoding="UTF-8" ?>
<xsl:stylesheet version="1.0" xmlns:oraxsl="http://www.oracle.com/XSL/Transform/java"
                xmlns:DVMFunctions="http://www.oracle.com/XSL/Transform/java/com.bea.wli.sb.functions.dvm.DVMFunctions"
                xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
                xmlns:BasicCredentialsUserFunction="http://www.oracle.com/XSL/Transform/java/com.bea.wli.sb.stages.functions.BasicCredentialsUserFunction"
                xmlns:tns="http://www.example.org" xmlns:oracle-xsl-mapper="http://www.oracle.com/xsl/mapper/schemas"
                xmlns:UUIDUserFunction="http://www.oracle.com/XSL/Transform/java/com.bea.wli.sb.stages.functions.UUIDUserFunction"
                xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                xmlns:ns0="http://xmlns.test.com/pcbpel/adapter/db/sp/DB_AccHierarchy_Pkg"
                xmlns:xsd="http://www.w3.org/2001/XMLSchema"
                xmlns:IsUserInRoleFunction="http://www.oracle.com/XSL/Transform/java/com.bea.wli.sb.stages.functions.IsUserInRoleFunction"
                xmlns:IsUserInGroupFunction="http://www.oracle.com/XSL/Transform/java/com.bea.wli.sb.stages.functions.IsUserInGroupFunction"
                xmlns:XrefFunctions="http://www.oracle.com/XSL/Transform/java/com.bea.wli.sb.functions.xref.XrefFunctions"
                exclude-result-prefixes="xsi oracle-xsl-mapper xsl xsd ns0 tns oraxsl DVMFunctions BasicCredentialsUserFunction UUIDUserFunction IsUserInRoleFunction IsUserInGroupFunction XrefFunctions">
  <oracle-xsl-mapper:schema>
    <!--SPECIFICATION OF MAP SOURCES AND TARGETS, DO NOT MODIFY.-->
    <oracle-xsl-mapper:mapSources>
      <oracle-xsl-mapper:source type="XSD">
        <oracle-xsl-mapper:schema location="DBSchema.xsd"/>
        <oracle-xsl-mapper:rootElement name="OutputParameters"
                                       namespace="http://xmlns.test.com/pcbpel/adapter/db/sp/DB_AccHierarchy_Pkg"/>
      </oracle-xsl-mapper:source>
    </oracle-xsl-mapper:mapSources>
    <oracle-xsl-mapper:mapTargets>
      <oracle-xsl-mapper:target type="XSD">
        <oracle-xsl-mapper:schema location="FinalSchema.xsd"/>
        <oracle-xsl-mapper:rootElement name="DBRowList" namespace="http://www.example.org"/>
      </oracle-xsl-mapper:target>
    </oracle-xsl-mapper:mapTargets>
    <!--GENERATED BY ORACLE XSL MAPPER 12.1.3.0.0(XSLT Build 140529.0700.0211) AT [FRI OCT 02 18:55:31 BRT 2015].-->
  </oracle-xsl-mapper:schema>
  <!--User Editing allowed BELOW this line - DO NOT DELETE THIS LINE-->
  <xsl:template match="/">
    <tns:DBRowList>
      <xsl:for-each select="/ns0:OutputParameters/ns0:H_DETAILS/ns0:Row">
        <tns:dbRow>
          <xsl:attribute name="nil" namespace="http://www.w3.org/2001/XMLSchema-instance"/>
          <xsl:if test="ns0:Column[@name='T_LEVEL']">
            <tns:T_LEVEL>
              <xsl:value-of select="ns0:Column[@name='T_LEVEL']"/>
            </tns:T_LEVEL>
          </xsl:if>
          <xsl:if test="ns0:Column[@name='T_CODE']">
            <tns:T_CODE>
              <xsl:value-of select="ns0:Column[@name='T_CODE']"/>
            </tns:T_CODE>
          </xsl:if>
          <xsl:if test="ns0:Column[@name='P_ID']">
            <tns:P_ID>
              <xsl:value-of select="ns0:Column[@name='P_ID']"/>
            </tns:P_ID>
          </xsl:if>
          <xsl:if test="ns0:Column[@name='P_NAME']">
            <tns:P_NAME>
              <xsl:value-of select="ns0:Column[@name='P_NAME']"/>
            </tns:P_NAME>
          </xsl:if>
          <xsl:if test="ns0:Column[@name='LAST_UPDATE_DATE']">
            <tns:LAST_UPDATE_DATE>
              <xsl:value-of select="ns0:Column[@name='LAST_UPDATE_DATE']"/>
            </tns:LAST_UPDATE_DATE>
          </xsl:if>
        </tns:dbRow>
      </xsl:for-each>
    </tns:DBRowList>
  </xsl:template>
</xsl:stylesheet>

示例输入(从 DB 适配器生成):

<?xml version="1.0" encoding="UTF-8" ?>
<OutputParameters xmlns="http://xmlns.test.com/pcbpel/adapter/db/sp/DB_AccHierarchy_Pkg">
  <H_DETAILS>
    <Row>
      <Column name="T_LEVEL" sqltype="NUMBER">1</Column>
      <Column name="T_CODE" sqltype="VARCHAR2">XYZ</Column>
      <Column name="P_ID" sqltype="NUMBER">13214</Column>
      <Column name="P_NAME" sqltype="VARCHAR2">XYZ_1 Limited</Column>
      <Column name="LAST_UPDATE_DATE" sqltype="TIMESTAMP">2015-07-01T09:21:27.901107+05:30</Column>
    </Row>
    <Row>
      <Column name="T_LEVEL" sqltype="NUMBER">2</Column>
      <Column name="T_CODE" sqltype="VARCHAR2">XYZ</Column>
      <Column name="P_ID" sqltype="NUMBER">156219</Column>
      <Column name="P_NAME" sqltype="VARCHAR2">XYZ_2 Limited</Column>
      <Column name="LAST_UPDATE_DATE" sqltype="TIMESTAMP">2015-07-02T09:21:27.901107+05:30</Column>
    </Row>
  </H_DETAILS>
</OutputParameters>

示例输出:

<?xml version = '1.0' encoding = 'UTF-8'?>
<tns:DBRowList xmlns:tns="http://www.example.org" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.example.org file:/home/mfelipe/jdeveloper/mywork/POC/SBProject/FinalSchema.xsd">
   <tns:dbRow xmlns:ns0="http://www.w3.org/2001/XMLSchema-instance" ns0:nil="">
      <tns:T_LEVEL>1</tns:T_LEVEL>
      <tns:T_CODE>XYZ</tns:T_CODE>
      <tns:P_ID>13214</tns:P_ID>
      <tns:P_NAME>XYZ_1 Limited</tns:P_NAME>
      <tns:LAST_UPDATE_DATE>2015-07-01T09:21:27.901107+05:30</tns:LAST_UPDATE_DATE>
   </tns:dbRow>
   <tns:dbRow xmlns:ns1="http://www.w3.org/2001/XMLSchema-instance" ns1:nil="">
      <tns:T_LEVEL>2</tns:T_LEVEL>
      <tns:T_CODE>XYZ</tns:T_CODE>
      <tns:P_ID>156219</tns:P_ID>
      <tns:P_NAME>XYZ_2 Limited</tns:P_NAME>
      <tns:LAST_UPDATE_DATE>2015-07-02T09:21:27.901107+05:30</tns:LAST_UPDATE_DATE>
   </tns:dbRow>
</tns:DBRowList>

【讨论】:

以上是关于如何编写 PL 以返回通过服务总线中的 DB 适配器自动生成强类型 XSD 的多行? (甲骨文 PL/SQL)的主要内容,如果未能解决你的问题,请参考以下文章

如何编写将部分传入消息传递给服务总线队列的 Azure HTTP 触发器函数?

DB2 和 PL/1:通过创建对适当游标的引用来避免冗余,使用它

如何执行目录中的所有 pl/sql 文件(install_db 脚本)

cmd中登录Oracle 显示ORA-12560: TNS: 协议适配器错误

在 Azure 数据工厂中完成活动后,如何向 Azure 服务总线发送消息

如何以不同的方式处理 PL/SQL 中的不同异常?