Chapter D - Fault Handling

Posted at 2010. 8. 12. 13:27 | Posted in Oracle Solution/PO Processing
반응형
Introduction

에러가 발생했을때 처리하는 법을 알아보겠습니다.



Test service unavailability

원격 에러(remote fault) 에 대한 에러 처리 정책을 정의하는 법을 알아봅시다.

먼저 validationForCC 웹서비스를 잠시 내려봅시다.

JDeveloper의 Application Server Navigator에서 validationForCC(Composite)를 내릴 수 있습니다.





※ Application Server Navigator가 안보인다면 View 메뉴에 있습니다.



※ EM(Enterprise Manager)에서도 내릴 수 있습니다.



이제 POProcessing를 테스트 합시다.

<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">
    <soap:Body xmlns:ns1="http://xmlns.oracle.com/ns/order">
        <ns1:PurchaseOrder>
            <ns1:CustID>1111</ns1:CustID>
            <ns1:ID>2222</ns1:ID>
            <ns1:productName>iPod shuffle</ns1:productName>
            <ns1:itemType>Electronics</ns1:itemType>
            <ns1:price>145</ns1:price>
            <ns1:quantity>30</ns1:quantity>
            <ns1:status>Initial</ns1:status>
            <ns1:ccType>Mastercard</ns1:ccType>
            <ns1:ccNumber>1234-1234-1234-1234</ns1:ccNumber>
        </ns1:PurchaseOrder>
    </soap:Body>
</soap:Envelope>

메시지 흐름 추적 실행을 해보면 에러가 발생했습니다.



getStatusByCC 웹서비스를 호출해야 하는데 웹서비스가 꺼져있어서 나는 에러 입니다.



remoteFault를 확인할 수 있습니다.





Add policy-based fault handler to do manual recovery

사용자 복구를 위한 기본 정의를 추가해봅시다.

아래 두 파일을 POProcessing 프로젝트 폴더에 복사합니다.

 



JDeveloper의 Application Navigator에서 새로고침을 누르면 복사된 파일이 나타납니다.



fault-policies.xml 파일은 에러에 따른 처리를 정의하는 파일이고, fault-bindings.xml 파일은 이 정의를 적용시키는 역할을 합니다.

fault-policies.xml 파일을 열어서 편집합니다.

Step D.2.2.2 주석 바로 아래에 아래의 내용을 추가합니다.

      <!-- Step D.2.2.2: Add your fault handler for remote fault here: -->
      <faultName xmlns:bpelx="http://schemas.oracle.com/bpel/extension" name="bpelx:remoteFault">
        <condition>
            <action ref="ora-human-intervention" />
        </condition>
      </faultName>

모두 저장 후 배치 합니다.

EM 에서 다시 테스트 후 메시지 흐름 추적 실행을 해보면 getStatusByCC는 에러가 나있고, approveLargeOrder(BPEL Process)는 실행 중인 것을 확인할 수 있습니다.

그리고 오류 메시지 부분에 복구라는 부분이 생겼습니다.



감사 추적을 보면 아래와 같은 부분이 나옵니다.



BPEL Process의 흐름을 보면 invokeCCStatusService 액티브에서 노란색 오로라를 뿜고 있습니다.

이 Invoke 부분에서 remote fault이 발생하여 사용자가 수동으로 처리를 해주기를 기다리고 있는 것입니다.



※ 여기서도 확인할 수 있습니다.



이제 validationForCC composite를 다시 살립시다.



위의 상태에서 "복구"를 누르거나 BPEL Process 추적 창에서 복구탭을 누르거나(방법은 좀 있는듯...) 해서 복구 탭으로 이동합니다.

에러 메시지 선택 후 복구 버튼을 클릭합니다.



예 클릭



정상적으로 에러가 없이 됐으면 아래와 같이 오류 메시지 목록에 아무것도 없습니다.



흐름 추적을 보변 2번째 복구(1번은 처음 실행시에 난 에러)에서 처리가 되어서 정상 처리가 끝까지 된 것을 확인할 수 있습니다.





Handle faults in BPEL

먼저 Database에서 작업을 하나 해줘야합니다. soademo/soademo로 들어가서 create_validate_cc.sql 스크립트를 실행해주세요.



CreditCardValidation Application - validationForCC Project - composite.xml 파일을 열어 편집합니다.

Database Adapter를 추가합니다.



Service name: validateCC



JNDI Name: eis/DB/soademoDatabase



Call a Stored Procedure or Function을 선택합니다.



Browse 버튼을 클릭합니다.



VALIDATECC를 선택합니다.



선택된 프로시저 확인 후 Next 클릭



Next



Finish



RouteReques(Mediator)와 validateCC(Database Adapter)를 연결합니다.



RouteReques(Mediator)를 더블 클릭하여 편집합니다.

getCreditValidation으로 가는 조건을 입력합니다.



not(starts-with($in.request/inp1:creditcardStatusRequest/inp1:CCNumber, '2'))

카드번호 맨 앞자리가 '2'로 시작하지 않는 조건입니다.



validateCC로 가는 조건도 입력합니다.

starts-with($in.request/inp1:creditcardStatusRequest/inp1:CCNumber, '2')

카드번호 맨 앞자리가 '2'로 시작하는 조건입니다.

validateCC로 가는 Transform 맵핑을 설정합니다.







validateCC에서 리턴받는 Transform 맵핑을 설정합니다.







모두 저장후 배치 합니다.

카드번호가 2로 시작하는 데이터를 입력합니다.

<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">
    <soap:Body xmlns:ns1="http://xmlns.oracle.com/ns/order">
        <ns1:PurchaseOrder>
            <ns1:CustID>1111</ns1:CustID>
            <ns1:ID>2222</ns1:ID>
            <ns1:productName>iPod shuffle</ns1:productName>
            <ns1:itemType>Electronics</ns1:itemType>
            <ns1:price>145</ns1:price>
            <ns1:quantity>30</ns1:quantity>
            <ns1:status>Initial</ns1:status>
            <ns1:ccType>Mastercard</ns1:ccType>
            <ns1:ccNumber>2234-1234-1234-1234</ns1:ccNumber>
        </ns1:PurchaseOrder>
    </soap:Body>
</soap:Envelope>

binding fault 에러가 발생했습니다.



BPEL 내에서 binding fault 에러를 제어 해봅시다.

approverLargeOrder(BPEL Process)를 편집합니다.



assignCCNumber(Assign) 액티브와 invokeCCStatusService(Invoke) 액티브 사이에 Scope 액티브를 추가합니다.



이름 부분을 더블 클릭 후 checkCC로 변경합니다.



Scope 액티브를 펼칩니다.



invokeCCStatusService(Invoke) 액티브를 Scope 액티브 안으로 넣습니다.



Scope 내의 아이콘중에 Add Catch Branch 아이콘을 클릭합니다.



추가된 Catch Branch를 펼칩니다.



Catch 아이콘을 더블 클릭합니다.



Browse Faults 버튼을 클릭합니다.



System Faults > bindingFault 을 선택합니다.



Fault Variable을 자동 생성합니다.



OK를 클릭합니다.



OK를 클릭합니다.



catch 영역 안에 Switch 액티브를 추가합니다.



<case> 의 조건을 입력합니다.



<case> 영역에 Assign 액티브를 추가합니다.



이름을 Assign_UNKNOWNCC로 변경 후 아이콘을 더블 클릭합니다.



From은 'UNKNOWN CC' 입력, To는 invokeCCStatusService_execute_OutputVariable - reply - creditcardStatus를 선택합니다.



<otherwise> 영역에 Throw 액티브를 추가합니다.



이름을 Throw_Binding_Fault로 변경 후 아이콘을 더블 클릭합니다.



Fault QName은 System Faults > bindingFault 선택 후 Fault Variable 오른쪽에 Browse Fault Variables.. 버튼을 클릭합니다.



catch 설정할 때 만들었던 FaultVar를 선택합니다.



OK 클릭



fault-policies.xml 파일을 열어 아래 내용을 추가합니다. (Step D.2.4.20 주석 아래에)

      <!-- Step D.2.4.20: Add your fault handler for binding fault here: -->
      <faultName xmlns:bpelx="http://schemas.oracle.com/bpel/extension" name="bpelx:bindingFault">
        <condition>
          <!-- Let the component handle this specific binding fault -->
          <test>$fault.code="20001"</test>
          <action ref="ora-rethrow-fault" />
        </condition>
      </faultName>

다시 카드번호가 '2'로 시작하는 데이터로 테스트 해보겠습니다.

getStatusByCC 서비스에서 에러는 났지만 approveLargeOrder(BPEL Process)는 정상적으로 완료 되었습니다.



감사 추적을 살펴보면 invokeCCStatusService 액티브에서 오류가 발생했고 catch 영역에서 에러를 처리했습니다.



BPEL에서 Scope는 Java의 try catch 역할도 합니다.



Using A Custom Java Fault Handler

이번에는 사용자가 직접 Java로 작성한 에러 처리를 적용해보겠습니다.

※ 에러처리에 사용할 예제 java 소스(jar)가 웹로직 서버 도메인의 lib 폴더에 있어야 합니다. 서버 실행중에 적용했다면 재시작!!

그리고 C:\po\log 폴더(웹로직 서버가 구동중인 컴퓨터입니다!!! 주의!)가 존재야 합니다.




fault-policies.xml 파일을 아래와 같이 수정합니다.

      <!-- Step D.2.4.20: Add your fault handler for binding fault here: -->
      <faultName xmlns:bpelx="http://schemas.oracle.com/bpel/extension" name="bpelx:bindingFault">
        <condition>
          <!-- Let the component handle this specific binding fault -->
          <test>$fault.code="20001"</test>
          <!-- <action ref="ora-rethrow-fault" /> -->
          <action ref="my-java-handler" />
        </condition>
      </faultName>

모두 저장 후 다시 배치. 카드 번호가 '2'로 시작하는 데이터로 테스트 해보겠습니다.

잘 처리 되었습니다.



soa console 에는 아래와 같이 출력됩니다.

MyFaultHanlder: Handle Fault
MyFaultHanlder: Properties
MyFaultHanlder: =================================================================
MyFaultHanlder: logFileName = myfaulthandler.log
MyFaultHanlder: logFileDir = c:\po\log

c:\po\log\myfaulthandler.log 파일을 열어보면 아래와 같습니다.

Fault Details
===============================================================
Fault Type ................ bpel
Poilcy ID ................. POProcessingFaults
Faulted Partner Link ...... getGreditCardStatus
Port Type ................. {http://oracle.com/sca/soapservice/CreditCardValidation/validationForCC/getStatusByCC}getGreditCardStatus



Handle Mediator Faults

Mediator에서 나는 에러를 처리해봅시다.

fault-policies,xml 을 아래와 같이 편집합니다. (Step D.2.6.1 주석 아래에 추가)

      <!-- Step D.2.6.1: Add your fault handler for mediator faults here: -->
      <faultName xmlns:medns="http://schemas.oracle.com/mediator/faults" name="medns:mediatorFault">
        <condition>
          <action ref="my-mediator-fault-handler" />
        </condition>
      </faultName>

Action을 하나 더 추가합니다. (Step D.2.6.2 주석 아래 추가)

      <!-- Step D.2.6.2: Add the Action definition for handling mediator fauls using custom java here:-->
      <Action id="my-mediator-fault-handler">
        <javaAction className="soatraining.faulthandling.MyFaultHandler"
            defaultAction="ora-terminate" propertySet="myMediatorProps">
            <returnValue value="OK" ref="ora-human-intervention" />
        </javaAction>
      </Action>

새로운 JavaAction에 사용되는 프로퍼티를 세팅합니다. (Step D.2.6.3 주석 아래 추가)

      <!-- Step D.2.6.3?: Add new property set for MyFaultHandler for logging Mediator faults here:-->
      <propertySet name="myMediatorProps">
        <property name="logFileName">mediator-faults.log</property>
        <property name="logFileDir">c:\po\log</property>
      </propertySet>

routePO(Mediator)를 수정합니다.



WriteApprovalResults 로 가는 라우팅을 병렬(Parallel)로 변경합니다.



Yes를 클릭합니다.



모두 저장 후 재배치합니다.

에러를 내기 위헤서 파일이 저장되는 폴더(C:\Temp)를 읽기 전용(쓰기 불가능)으로 만들겠습니다.

윈도우(Windows)의 경우
  - 읽기전용 설정
    > attrib +r c:\temp
  - 읽기전용 해제
    > attrib -r c:\temp
리눅스(Linux)의 경우
  - 읽기전용 설정
    $ chmod -w /temp
  - 읽기전용 해제
    $ chmod +w /temp 

이제 EM(Enterprise Manager)에서 테스트를 해봅시다.

<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">
    <soap:Body xmlns:ns1="http://xmlns.oracle.com/ns/order">
        <ns1:PurchaseOrder>
            <ns1:CustID>1111</ns1:CustID>
            <ns1:ID>2121</ns1:ID>
            <ns1:productName>Bluetooth Headset</ns1:productName>
            <ns1:itemType>Electronics</ns1:itemType>
            <ns1:price>49.99</ns1:price>
            <ns1:quantity>1</ns1:quantity>
            <ns1:status>Initial</ns1:status>
            <ns1:ccType>Mastercard</ns1:ccType>
            <ns1:ccNumber>8765-8765-8765-8765</ns1:ccNumber>
        </ns1:PurchaseOrder>
    </soap:Body>
</soap:Envelope>

메시지 흐름 추적을 보면 에러가 났습니다.



감사 추적을 보면 WriteApprovalResults.Write 부분에서 에러가 난것을 볼 수 있습니다.



C:\Temp 폴더의 읽기 전용을 해제 한뒤, 복구를 해봅시다.



예 클릭



복구 작업이 성공적으로 완료되었습니다.



다시 메시지 흐름 추적을 보면 재시도한 흔적(?)을 확인할 수 있습니다.







반응형
//

Transform에서 for-each 주의사항!

Posted at 2010. 4. 29. 14:26 | Posted in Oracle Solution/BPEL
반응형
일반적으로 트랜스폼(Transform)에서 여러개의 데이터일 경우 for-each 컴포넌트를 이용하여 돌릴 수 있습니다.

하지만 내가 의도하지 않는 결과가 나올 때가 있습니다. ㅠ_ㅠ



아래는 입력과 출력의 XML 스키마입니다.



단순히 입력 받은 그대로를 출력하는 BPEL 입니다.



트랜스폼으로 처리 했습니다.

<sample> 태그의 수만큼 루프가 돌게 됩니다.





<soap:Envelope xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">
  <soap:Body xmlns:ns1="http://www.example.org">
    <ns1:request>
      <ns1:sample>  <!-- 1개 -->
        <ns1:seq>1</ns1:seq>
        <ns1:id>1</ns1:id>
        <ns1:data>data_1</ns1:data>
      </ns1:sample>
      <ns1:sample>  <!-- 2개 -->
        <ns1:seq>2</ns1:seq>
        <ns1:id>2</ns1:id>
        <ns1:data>data_2</ns1:data>
      </ns1:sample>
    </ns1:request>
  </soap:Body>
</soap:Envelope>

위와 같은 데이터를 입력하게 되면 아래와 같은 데이터가 나오게 됩니다.

<env:Envelope xmlns:env="http://schemas.xmlsoap.org/soap/envelope/"
  xmlns:wsa="http://www.w3.org/2005/08/addressing">
  <env:Header>
    <wsa:MessageID>urn:231BE770534E11DFBFC6CF3D19573BDE</wsa:MessageID>
    <wsa:ReplyTo>
      <wsa:Address>http://www.w3.org/2005/08/addressing/anonymous</wsa:Address>
    </wsa:ReplyTo>
  </env:Header>
  <env:Body>
    <response xmlns:ns0="http://www.example.org" xmlns="http://www.example.org">
      <ns0:sample>
        <ns0:seq>1</ns0:seq>
        <ns0:id>1</ns0:id>
        <ns0:data>data_1</ns0:data>
      </ns0:sample>
      <ns0:sample>
        <ns0:seq>2</ns0:seq>
        <ns0:id>2</ns0:id>
        <ns0:data>data_2</ns0:data>
      </ns0:sample>
    </response>
  </env:Body>
</env:Envelope>

하지만!! 아래와 같이 의도하지 않은 결과가 나올때가 있습니다. ㅠ_ㅠ

갯수는 제대로 나왔는데 데이터가 1첫째의 데이터만 계속 반복해서 나옵니다.

<env:Envelope xmlns:env="http://schemas.xmlsoap.org/soap/envelope/"
  xmlns:wsa="http://www.w3.org/2005/08/addressing">
  <env:Header>
    <wsa:MessageID>urn:71B1A920534D11DFBFC6CF3D19573BDE</wsa:MessageID>
    <wsa:ReplyTo>
      <wsa:Address>http://www.w3.org/2005/08/addressing/anonymous</wsa:Address>
    </wsa:ReplyTo>
  </env:Header>
  <env:Body>
    <response xmlns:ns0="http://www.example.org" xmlns="http://www.example.org">
      <ns0:sample>
        <ns0:seq>1</ns0:seq>
        <ns0:id>1</ns0:id>
        <ns0:data>data_1</ns0:data>
      </ns0:sample>
      <ns0:sample>
        <ns0:seq>1</ns0:seq>
        <ns0:id>1</ns0:id>
        <ns0:data>data_1</ns0:data>
        </ns0:sample>
    </response>
  </env:Body>
</env:Envelope>

Mapper 파일을 Source 모드로 보면 정상적인 for-each 가 적용되면 아래와 같이 됩니다.

  <xsl:template match="/">
    <ns0:response>
      <xsl:for-each select="/ns0:request/ns0:sample">
        <ns0:sample>
          <ns0:seq>
            <xsl:value-of select="ns0:seq"/>
          </ns0:seq>
          <ns0:id>
            <xsl:value-of select="ns0:id"/>
          </ns0:id>
          <ns0:data>
            <xsl:value-of select="ns0:data"/>
          </ns0:data>
        </ns0:sample>
      </xsl:for-each>
    </ns0:response>
  </xsl:template>

하지만 정상적으로 for-each가 안되는 소스를 보면 아래와 같이 되어있습니다.

  <xsl:template match="/">
    <ns0:response>
      <xsl:for-each select="/ns0:request/ns0:sample">
        <ns0:sample>
          <ns0:seq>
            <xsl:value-of select="/ns0:request/ns0:sample/ns0:seq"/>
          </ns0:seq>
          <ns0:id>
            <xsl:value-of select="/ns0:request/ns0:sample/ns0:id"/>
          </ns0:id>
          <ns0:data>
            <xsl:value-of select="/ns0:request/ns0:sample/ns0:data"/>
          </ns0:data>
        </ns0:sample>
      </xsl:for-each>
    </ns0:response>
  </xsl:template>

현재 JDeveloper 의 버그 같지만 이런 에러(?)가 안나게 하는 방법은 맵핑할 때 자동 맵핑(Auto Map)을 이용하거나, for-each 컴포넌트를 먼저 적용 한 후 선을 연결하면 됩니다.

선을 먼저 연결하고 for-each 컴포넌트를 적용하면 원하는 결과가 안나올 수 있습니다.

반응형

'Oracle Solution > BPEL' 카테고리의 다른 글

프로젝트 구조가 이상해질 때!!!  (0) 2010.04.27
Database Adapter  (0) 2010.04.19
//

프로젝트 구조가 이상해질 때!!!

Posted at 2010. 4. 27. 17:52 | Posted in Oracle Solution/BPEL
반응형
JDeveloper 11g

후... 열심히(진짜루..) 일하고 있는 갑자기 프로젝트 구조가 갑자기 아래와 같이 바뀔 때가 있습니다. ㄷㄷ!



이런
같은 경우가 ㅠㅠ

첨에는 당황해서 지우구 다시 만들었는데(ㅠ_ㅠ) 계속 이러길래 이것저것 해본 결과!!

프로젝트 정보가 좀 깨져서 SOA의 정보가 날라간거 같습니다. 왜 그런지는 몰름! ㅠ_ㅠ



예] 프로젝트 경로 : D:\workspace\samyang\EDAS_SYC\EDAS_SYC_F0006

먼저 프로젝트에서 마우스 오른쪽 버튼 클릭 후 Project Properties... 선택



Project Source Paths에서 "{프로젝트 경로}\SCA-INF\src" 경로를 하나 더 추가해 줍시다!



SOA Content 에서 SOA Directory Name 부분에 {프로젝트 경로}를 설정합니다.



Resources 에서 기존에 지정되어있는 경로를 삭제 합니다.





Waaagh!! 다시 원상태로 돌아왔습니다. ㅠ_ㅠ



하지만 배치(Deploy)하려고 보니까 배치 정보가 없군염 ㅠㅠ

새 배치 프로필을 생성 합시다



SOA-SAR File 선택



보통 프로젝트 이름을 사용 하죠~



그냥 OK 클릭



Project Properties 창이 다시 열리고 Deployment 에 프로필이 생겼습니다.



이제 이상해지기 전의 프로젝트 상태로 복구(?) 되었습니다. 휴우~
반응형

'Oracle Solution > BPEL' 카테고리의 다른 글

Transform에서 for-each 주의사항!  (1) 2010.04.29
Database Adapter  (0) 2010.04.19
//

Database Adapter

Posted at 2010. 4. 19. 11:54 | Posted in Oracle Solution/BPEL
반응형
프로젝트를 하게 되면 가장 많이 사용하게 되는 것이 데이터베이스(이하 디비)입니다.

BPEL 에서는 디비에 CRUD[각주:1]를 하려면 Database Adapter(이하 DbApapter)라는 것을 사용합니다.

하지만.... 그냥 쓰면 되는 것은 아니고... DbAdapter를 하나 추가 하기 위해서 미리 설정을 해줘야 하는 것이 있습니다.

웹로직 콘솔에서 데이터소스(datasource)를 설정하고, 그 데이터소스를 사용하는 DbAdapter를 추가해야 실제로 배치(deploy) 했을 때 사용 가능합니다.



간단한 샘플로 어떻게 해야되는지 알아봅시당당당~

웹서비스로 유저 정보를 입력하면 Oracle과 Sql Server 2005 에 각각 Insert 하는 웹서비스(?)를 만들어보겠습니다.



스키마 파일입니다.

  

입력되는 필드는 아래와 같습니다.



※ 아래 예제를 하시려면 SOA Suite 설치 및 프로젝트 생성, 배치는 할줄 알아야 합니둥~ -_-;;



라이브러리(jar) 세팅

웹로직(Oracle WebLogic)에서 기본적인 jdbc 드라이버는 포함되어 있지만 Mssql용 jdbc 드라이버는 포함되어있지 않습니다.

그래서 jdbc 드라이버를 추가해야 합니다. 많은 방법이 있지만 저는 {도메인홈}\lib 디렉토리에 넣어두겠습니다.





웹로직 서버를 시작(재시작)핼 때 콘솔화면을 보면 아래와 같이 sqljdbc4.jar 파일이 클래스 패스에 포함된 걸 볼 수 있습니다.





데이터베이스 리소스(datasource) 생성

※ 데이터소스 생성에 관해서는 제 블로그의 다른 글에도 많이 있을 겁니다. ㄷㄷ

먼저 Oracle 10g 데이터소스를 생성하겠습니다.

웹로직 콘솔로 들어갑니다. 도메인 구조(메뉴)에서 서비스 - JDBC - 데이터 소스를 선택합니다.



새로 만들기를 클릭합니다.



이름, JNDI[각주:2] 이름, 데이터베이스 유형, 데이터베이스 드라이버를 설정합니다.

여기서 중요한게 JNDI 이름입니다. 이 이름으로 찾아서 리소스를 사용합니다.

다음을 클릭합니다.



트랜잭션을 설정합니다. XA 지원되는 jdbc 드라이버는 패스 됩니다. ㅋㅋ



디비 접속 정보를 설정합니다. 입력 후 다음을 클릭합니다.



구성 테스트를 클릭합니다.



"접속 테스트가 성공했습니다." 메시지가 나와야 정상입니다.

다음을 클릭합니다.



대상을 선택합니다. 꼭 대상을 선택해 줘야합니다.

soa_server1을 체크합니다.




SQL Server 20005 데이터 소스를 생성하겠습니다.

데이터 소스 목록에서 sampleOracle이 추가된 것을 볼 수 있습니다.

새로 만들기 클릭



이름, JNDI 이름, 데이터베이스 유형, 데이터베이스 드라이버를 설정합니다.

기본적으로 Mssql은 XA[각주:3]가 아니므로 XA가 아닌 드라이버를 선택했습니다.

다음 클릭



트랜잭선 설정입니다.

글로벌 트랜잭션 지원
체크 해제 합니다. 다음 클릭



접속 정보를 설정합니다.



구성 테스트를 클릭합니다.



다음을 클릭합니다.



대상을 soa_server1로 설정합니다.



sampleMssql(jdbc/sampleMssql), sampleOracle(jdbc/sampleOracle) 이 추가되었습니다.




DbAdapter 설정

BPEL 에서 디비에 접근하려면 DbAdapter 를 통해서 해야합니다.

왼쪽 트리 메뉴에서 배치를 클릭합니다.



배치 목록에서 DbAdapter를 클릭합니다.



구성 탭을 클릭합니다.



아웃바운드 접속 풀 탭을 클릭합니다.



목록 테이블에서 새로 만들기를 클릭합니다.



javax.resource.cci.ConnectionFactory를 선택하고 다음을 클릭합니다.



JNDI 이름에 eis/DB/sampleOrcle을 입력합니다.

이 JNDI 이름이 JDeveloper 에서 사용할 이름입니다. 왕중요!!



처음 설정할 때 배치 계획 파일을 어디에 저장할 것인지 물어봅니다.

저는 {SOA홈}\soa\DBPlan 디렉토리에 저장하겠습니다. (DBPlan 디렉토리는 미리 만들었습니다.)

DBPlan 을 선택합니다.



확인을 클릭합니다.



정상 처리 메세지를 확인하시고, 배치 계획 경로도 확인해주세요~




다시 배치 - DbAdapter - 구성 - 아웃바운드 접속 풀 까지 이동합니다.

새로 만들기 클릭



그거.. 선택 후 다음 클릭



JNDI 이름으로 eis/DB/sampleMssql 입력합니다.



또! 다시 배치 - DbAdapter - 구성 - 아웃바운드 접속 풀 까지 합니다.

목록 테이블에 두개가 추가된 것을 볼 수 있습니다.

오라클꺼 먼저 설정 해봅시다.

javax.resource.cci.ConnectionFactory 앞에 +를 눌러 확장합니다.

eis/DB/sampleOracle을 선택합니다.



속성 테이블에서 맨 아래 xaDataSourceName의 속성값 부분 공간을 클릭합니다.



그러면 input이 생겨납니다. 우와왕~

여기에 jdbc/sampleOracle을 입력한 후 엔터를 입력합니다.



저장 클릭



잘 저장 되었다고 나옵니다. ㄷㄷ



이번에는 eis/DB/sampleMssql 클릭



현재 Mssql은 XA가 아니므로 dataSourceName에 jdbc/sampleMssql을 입력합니다.

platformClassName
을 변경해야 합니다. 현재 입력되어있는 속성값은 딱 봐도 Oracle10 용임을 알 수 있습니다.



oracle.toplink.platform.database.SQLServerPlatform 을 입력합니다.

다른 platformClassName은 아래 링크를 참조해주세요~

http://download.oracle.com/docs/cd/E12839_01/integration.1111/e10231/adptr_db.htm#CHDDDJAG



저장을 클릭합니다.



배치 목록에서 DbAdapter를 선택(체크박스 선택)하고 갱신을 클릭합니다.



완료를 클릭합니다.



성공적으로 갱신되었다는 메세지 확인



이제 JDeveloper 에서 디비 어댑터(Database Adapter)를 사용할 준비가 되었습니다.

JDeveloper에서 먼저 설정하고 콘솔에서 데이터소스와 DbAdapter를 설정해도 됩니다.

위에서 설정한 데이터소스와 DbAdapter는 실제 어플리케이션이 배치(Deploy)되서 돌아갈 때 사용하니까요~


Working JDeveloper

Database Addpater를 External Refereneces 영역에 드래그 앤 드랍 합니다.



서비스 이름(Service Name)을 지정합니다. 저는 그냥 테이블 이름을 했습니다.



현재 JDeveloper에서 사용할 접속 정보를 설정합니다.

커낵션(Connection) 오른쪽에 데이터베이스 커낵션 생성 버튼(녹색 +)을 클릭합니다.



커낵션 정보를 입력 후 Test Connection 버튼을 클릭했을 때 "Success!" 가 나와야 합니다.

OK 버튼을 클릭합니다.



커낵션을 추가(한번 추가했으면 앞으로는 선택하겠죠..)했을때의 모습니다.

아래 JNDI Name 이 eis/DB/{커낵션명} 으로 자동 작성되어있는 걸 볼 수 있습니다.

파란선을 기준으로 위쪽은 JDeveloper에서 설정할떄 쓰는 커낵션 정보이고, 아래 JNDI Name이나 Data Source는 실제 배치 되었을 때 참조하는 리소스 입니다.

그러므로 JDeveloper에서의 접송 정보와 위에서 했던 DataSource, DbAdpater 설정이 일치해야 합니다.

JNDI Name 이 위에서 설정한 DbAdapter 이름이어야 한다는게 감이 오시졈? ㄷㄷ

Next 버튼을 클릭합니다.



데이터베이스에 접속해서 뭘 할지 설정합니다. -_-

딱 보면 뭔지 알 수 있습니다.

Poll for New or Changed Records in a Table 은 테이블을 지속적으로 읽어들여 변화가 있을 떄 이벤트가 일어납니다.

Execute Pure SQL 은 사용자 쿼리 입니다.



테이블을 선택합니다. Import Tables 클릭



Query 버튼을 클릭하면 Available 영역에 테이블 목록이 조회됩니다.

테이블을 선택하고 오른쪽으로 가는 화살표 버튼을 클릭하여 선택합니다.



테이블이 선택된 것을 확인하시고 Next 버튼을 클릭합니다.



※ 임포트한 테이블에 기본키(Primary Key)가 없을 경우 기본키를 지정하는 단계가 나옵니다.

관계(Relationships)를 설정합니다. 일단 그런거 없으니 패스



사용할 필드를 설정합니다. 전 다 쓸겁니다. Next 클릭



고급 설정입니다.

시퀀스 설정(Sql Server의 Identity도 되더군용) 이라든지 이것 저것 설정할 수 있습니다.

그냥 Next 클릭 ㄷㄷ



Finish 클릭



Database Adapter(Oracle 꺼)가 하나 생성되었습니다.

이제 Mssql 꺼를 하나 더 만듭시다~





Mssql 커낵션을 추가합시다.



오라클이 아닌 다른 디비는 jdbc 드라이버를 설정해줘야 합니다.

Library 오른쪽이 Browse 클릭



New 클릭



Add Entry 클릭



jdbc 드라이버를 선택합니다.



라이브러리 이름(Libary Name)을 지정하고 OK 클릭



라이브러리가 추가 되었습니다. OK 클릭



Test Connection 버튼을 클릭하면 "Success!" 메세지가 나와야합니다.

만약 라이브러리를 설정 한하면 클래스가 없다느니 뭐라고 에러가 날겁니다. ㅎㅎ



다음 단계는 위의 Oracle 설정과 같습니다.





이제 요청(request) XML 구조를 만들어 봅시다.

xsd 폴더에서 마우스 오른쪽 버튼 클릭 후 New 클릭



All Technologies 탭에서 General - XML 에서 XML Schema를 선택합니다.



파일 이름(File Name)을 입력합니다.

보통 xsd 폴더에 저장하니 폴더도 확인 (꼭 xsd 폴더에 없어도 됩니당)



처음 만들게 되면 "exampleElement" 엘리먼트 하나만 만들어져 있습니다.

exampleElement 선택 후 오른쪽 컴포넌트 팔레트(Component Palette)에서 시퀀스(sequence)를 클릭합니다.

그러면 시퀀스가 하나 추가되고 추가된 시퀀스가 선택되어 있습니다.



엘리먼트(element)를 클릭합니다.

그러면 시퀀스 다음에 엘리먼트가 추가됩니다.



전에 추가된 시퀀스를 선택하고 다시 엘리먼트를 클릭합니다.



이렇게 엘리먼트 7개를 추가합니다.

뭐 드래그를 해도 되고 소스 편집을 해도 되고 방법은 많으니 직접 해보세용~



추가된 엘리먼트를 선택 후 오른쪽 아래에 속성 부분에서 이름(name)을 변경하시거나 엘리먼트를 더블클릭해서 이름을 바꿉니다.



그 후 다시 속성 부분에서 타입(type)을 입력합니다.



아래와 같이 7개의 엘리먼트를 완성합니다.

저장합니다.



Mediator 컴포넌트를 Components 영역에 드래그 앤 드랍합니다.



이름(Name)을 입력하고 템플릿(Template)은 One Way Interface(단방향)를 선택합니다.

Input은 방근전 만들었던 input을 선택합니다. 돋보기 버튼 클릭



Project Schema Files - input.xsd - exampleElement 를 선택합니다.



OK 클릭



RouteUser(Mediator)와 User(Database Adapter)를 연결합니다.



RouteUser와 TblUser를 연결합니다.



RouteUser(Mediator)를 더블 클릭하여 수정합니다.



아래에 보면 두개의 라우팅(Static Routing)이 있습니다.

각각 User와 UblUser로 가는 겁니다.

첫번째. User로 가는 데이터를 설정합시다. Transform Using 오른쪽의 맵핑 설정 버튼을 클릭하빈다.



Create New Mapper File을 선택합니다.



아래와 같이 선을 연결합니다.

그리고 String Functions 그룹에서 concat 컴포넌트를 가운데 영역에 드래그 앤 드랍합니다.



inp1:zipcode 와 inp1:address 를 차례로 concat 컴포넌트에 연결하고 top:addr 과 concat 컴포넌트와 연결합니다.

concat 컴포넌트를 더블클릭하여 수정합니다.



Add 버튼을 이용하여 파라메터(parameter)를 추가 한 후 공백(" ")을 넣습니다.



Date Functions 그룹에 current-dateTime 컴포넌트를 가운데 영역에 드래그 앤 드랍한 후, top:wdate 와 연결합니다.



이번에는 두번쨰 연결인 TblUser 와의 연결을 설정합시다.







모두 저장합니다.



배치(Deply) 및 테스트

배치 합니다.



테스트 합시다.



입력 후 오른쪽 아래 웹 서비스 테스트를 클릭합니다.



성공(?) 했습니다.

메시지 흐름 추적 실행을 클릭합니다.



RouteUser 를 거쳐서 User, TblUser 로 실행된 것을 볼 수 있습니다.

RouteUser를 클릭합니다.



Mediator에서 무슨일이 일어났는지 볼 수 있습니다.

페이로드를 확장하면 이동하는 데이터(XML)를 볼 수 있습니다.



데이터베이스를 확인해보면 입력한 정보가 각각 테이블에 들어가 있습니다. 와아~ -_-v





  1. C: Create, R: Retrieve(Read), U: Update, D: Delete [본문으로]
  2. Jana Naming and Directory Interface: 이름을 사용해서 객체나 객체에 대한 레퍼런스를 발견할 수 있는 방법을 제공한다. 어플리케이션 개발자는 네이밍 서비스에 이름을 사용하여 객체를 바인딩함으로써 객체에 이름을 부여하게 된다. 이후 코드에서는 글 이름을 사용하여 바인딩 된 객체를 참조할 수 있게 된다. [본문으로]
  3. XA방식의 트랜잭션이란 여러개의 데이터베이스, JMS, 또는 그 외의 리소스들간의 트랜잭션을 보장하는것을 말하고 NonXA방식이란 일반적인 한개의 데이터베이스에서 관리되는 트랜잭션을 말한다. [본문으로]
반응형

'Oracle Solution > BPEL' 카테고리의 다른 글

Transform에서 for-each 주의사항!  (1) 2010.04.29
프로젝트 구조가 이상해질 때!!!  (0) 2010.04.27
//

Chapter C - Unit Testing

Posted at 2010. 3. 2. 17:33 | Posted in Oracle Solution/PO Processing
반응형
Introduction

SOA composite 레벨에서 테스트를 할 수 있습니다.

아래 4가지 유닛 테스트를 해봅시다.

1. An inbound message for receivePO.
2. A simulation of a callback message returned by as service
3. An assertion to verify the order status at completion
4. An assertion the will always fail for the input we provided in this testcase



Create the Unit Test

프로젝트 - SOA Content - testsuites 에서 마우스 오른쪽 버튼 클릭 후 Create Test Suite 클릭




이름은 logic Test로 입력




생성될 파일 이름을 TestDelivery로 입력




Exposed Services와 External Reference 영역이 노란색으로 변경되었습니다.

Return to SOA composite diagram 버튼을 누르면 진짜 composite 화면으로 갑니다.





Set the inbound message

reveivePO를 더블클릭하거나 오른쪽 버튼 클릭 후 Create Initiate Messages를 클릭합니다.






request Part의 값을 입력합니다.




receivePO 그림에 파란색 화살표가 생겼습니다.




이번에는 getCreditCarStatus에서 approveLargeOrder로 리턴(return)이 항상 VALID가 나오게 해봅시다.

approveLargeOrder(BPEL Process)와 getCreditCardStatus(Web Service)를 연결하는 선을 더블 클릭합니다.



Emulates 탭에서 Emulate를 추가합니다.




Generate Sample 버튼을 클릭하면 아웃풋의 샘플이 나옵니다.

태그 사이의 값을 VALID로 수정합니다.




OK를 클릭합니다.




선이 변경되었습니다.





Set the assertion for success

routePO 에서 WriteApprovalResults 로 갈 때 상태(status) 내가 지정한 값과 같은지 검증합니다.

routePo(Mediator)와 WriteApprovalResults(File Adapter)를 연결하는 선을 더블 클릭합니다.




Asserts 탭에서 Assert를 추가합니다.




Assert Target 오른쪽의 Browse 버튼을 클릭합니다.




status를 선택합니다.




값을 approved 로 입력합니다.





Set the assertion for failure

이번에는 배송 처리할 때 고객ID(CustomerID)가 내가 지정한 값과 일치하는지 검증해봅시다.

approveLargeOrder와 FullfillementProcess를 연결하는 선을 더블 클릭합니다.




Assert를 추가합니다.




값은 9999를 입력합니다.




OK 클릭





Deploying the Application

배치합니다. 이번에는 혹시몰라 1.1 버젼으로 배치했습니다. ㄷㄷ





Testing the application

프로젝트(POProcessing [1.1]) 선택 후 단위 테스트 탭을 선택 후 실행을 클릭합니다.




테스트 실행 이름을 입력합니다.




테스트 실행의 결과 상태가 "실행 중"이면 테스트 상태 새로 고침을 버튼 클릭해서 상태를 갱신합니다.




상태가 "실패"로 나왔습니다.




아래쪽 검증 세부 정보를 보면 검증 결과가 나온 것을 확인할 수 있습니다.







반응형
//
반응형
Introduction





Surfacing binding properties at the composite level

File Adapter를 직접 수정하지 않고 프로퍼티만 설정해서 생성될 파일의 이름을 바꿔봅시다.

WriteApproval 레퍼런스(File Adapter)를 선택합니다.




레퍼런스(또는 Property Inspector) 패널에서 Binding Properties를 추가합니다.




Name은 FileNamingConvention을 선택하고 Value는 test_%SEQ%.xml을 입력합니다.




composite.xml을 Source 보기로 보면 아래와 같이 프로퍼티가 추가되었습니다.





Generating and validating a deployment plan

여치 계획 파일을 만들어서 배치할 때 여러가지 설정을 적용(변경)한 상태로 배치를 할 수 있습니다.

composite.xml 파일에서 마우스 오른쪽 버튼 클릭 후 Generate Config Plan을 클릭합니다.




생성할 파일 이름을 POProcessing_dev_cfgplan.xml 으로 설정 후 OK 클릭




아래의 위치에 텍스트를 추가합니다.

File Adapter 의 출력 파일명을 변경합니다.




아래의 위치에  텍스트를 추가합니다.

파일을 쓸 디렉토리의 위치를 변경합니다.




모두 저장 후 composite.xml 파일에서 오르쪽 버튼 클릭 후 Validate Config Plan을 클릭합니다.




configuration plan 파일을 선택합니다.




어떻게 바뀔 것인지 로그가 나옵니다.




서버에 디플로이 할때 아래와 같이 플랜 파일을 선택할 수 있습니다.





Testing

테스트를 해보면 주문 파일이 설정 플랜에서 설정한 대로 파일이 생성되는 것을 확인할 수 있습니다.



반응형
//
반응형
Introduction

배송정보를 JMS 큐(Queue)에 등록하겠습니다.

Chapter 1 에서 JMS 관련 서버 설정이 되어있어야 합니다.



Add the JMS adapters

JMS Adapter를 External References 영역에 추가합니다.



Name : JMS_USPS



Oracle Weblogic JMS를 선택합니다.



서버 커넥션을 선택합니다.



Define from operation and schema (specified later)를 선택합니다.



Produce Message를 선택합니다.



Browse... 를 클릭합니다.



demoFulfillmentQueue (queue)를 선택합니다.



JNDI Name : eis/QUeue/demo



browse 버튼을 클릭합니다.



fulfillment.xsd - Fulfillment 를 선택합니다.



Next 클릭



Finish 클릭




위와 같은 방법으로 JMS_UPS, JMS_FedEx를 추가로 만들어줍니다.

다 만든 후 FilfillmentProcess(BPEL Process)와 연결합니다.





Invoke the services from BPEL

FulfillmentProcess(BPEL Process)를 더블 클릭하여 편집합니다.



FulfillmentProcess(Business Rule 액티브) 아래에 Switch 액티브를 추가합니다.



Case를 2개 더 추가합니다.



<case> 부분을 더블 클릭합니다.



General 탭에서 Name은 USPS를 입력합니다.

Expression은 아래와 같이 입력합니다. (빌더를 이용하세요)



두번째 <case> 도 더블 클릭해서 UPS를 입력합니다.



세번째 <case>는  FedEx



<case USPS> 안에 Invoke 액티브를 추가합니다.




추가한 Invoke 액티브와 JMS_USPS 컴포넌트를 연결합니다.



이름은 Invoke_USPS를 입력하고, Input Variables 를 기본으로 추가합니다.



UPSFedEx도 마찬가지로 추가합니다.



Assign 액티브를 <case USPS> 영역 안의 Invoke_USPS 위에 추가합니다.



더블 클릭하여 편집합니다.



General 탭에서 이름을 Assign_USPS로 입력합니다.



Copy Operation을 아래와 같이 추가합니다.

UPSFedEx도 똑같이 추가합니다. (그래도 To는 틀려야겠죠? -_-)



<otherwish> 부분은 삭제합니다.



완성되었습니다.





Deploying and testing the application

배치 후 테스트 해보면 마지막에 JMS로 데이터가 간 것을 확인할 수 있습니다.



WebLogic Console로 접속해서 서비스 - 메시징 - JMS 모듈로 이동합니다.



SOAJMSModule을 클릭합니다.



demoFulfillmentQueue를 클릭합니다.



모니터링 탭에서 SOAJMSModule! demoFulfillmentQueue를 체크 후 메시지 표시를 클릭합니다.



들어온 메세지를 확인합니다.



데이터를 확인합니다.







반응형
//

Chapter 8 - Adding Fulfillment

Posted at 2010. 3. 2. 16:55 | Posted in Oracle Solution/PO Processing
반응형
Introduction

배송을 처리하는 BPEL Process 와 Business Rule 를 추가하여 봅시다.

1000원 이하는 USPS, 1000~5000원은 UPS, 5000원 초과는 FedEx 로 배송 처리합니다.



Add a BPEL Process and a Business Rule


BPEL Process 를 Components 영역에 추가합니다.




Name : FulfillmentProcess
Template: Asynchronous BPEL Process
Service Name : fulfillmentprocess_client
Expose as a SOAP service 체크 해제

Input 오른쪽의 찾기 버튼 클릭




Import 클릭






파일을 불러옵니다.






Fufillment 를 선택합니다.




Output 은 CarrierSelection 를 선택합니다.




Input 과 Output 확인 후 OK 클릭




Business Rule 컴포넌트를 Components 영역에 추가합니다.




Genral 탭에서

Name : FulfillmentRules
Package : fulfillmentrules

Input 을 추가합니다.




Fulfillment를 선택합니다.




이어서 Output을 추가합니다.




CarrierSelection을 선택합니다.



Advanced 탭에서 Service Name에 OracleRulesFulfillment 입력 후 OK를 클릭합니다.



approveLargeOrder와 FulfillmentProcess를 연결합니다.




FulfillmentProcess와 FulfillmentRules를 연결합니다.





Define the Business Rule

FulfillmentRules(Business Rule) 컴포넌트를 더블클릭하여 편집합니다.





왼쪽 메뉴에서 Bucketsets을 클릭합니다.



List of Ranges 를 추가합니다.



아이콘을 더블클릭하여 편집창을 띄웁니다.



Name은 OrderTotal, Data Type는 double 선택. 녹색 플러스 버튼을 클릭하여 아래와 같이 입력합니다.



왼쪽 메뉴에서 Facts를 클릭 한 후, 목록에서 FulfillmentType을 수정합니다.



total의 Bucketset을 OrderTotal로 설정합니다.



왼쪽 메뉴에서 Ruleset_1을 선택 후 Create Decision Table을 클릭합니다.



<insert condition> 부분을 클릭합니다.



<edit condition> 부분을 더블 클릭하고 FulfillmentType.total을 선택합니다.



R1 셀을 클릭 후 small을 선택합니다.



Rule을 추가합니다.



R2 셀에 large,를 선택합니다.

다시 Rule 추가 후 R3 셀에 xtralarge를 선택합니다.



Actions 부분에 Assert를 추가합니다.



assert new를 더블 클릭합니다.




Facts는 CarrierSelection을 선택하고 Parameterized를 체크합니다. 그리고 Always Selected를 체크합니다.

OK를 클릭합니다.



small의 액션에 CarrierValue.UPS를 선택합니다.



large에는 CarrierValue.UPS, xtralarge는 CarrierValue.FED_EX를 선택합니다.





Define the BPEL Process

FulfillmentProcess(BPEL Process)를 더블 클릭해서 편집합니다.



Business Rule
액티브를 추가합니다.



이름은 FulfillmentProcess를 입력, Dictionary는 FilfillmentRules를 선택합니다.

Assign Input Facts 탭에서 Assign을 추가합니다.



아래와 같이 선택합니다.



Assign Output Facts 탭에서 Assign을 생성합니다.





OK를 클릭합니다.





Invoke FulfillmentProcess

approveLargeOrder(BPEL Process를 더블 클릭합니다.



Invoke 액티브를 ManualPOApproval_1이 있는 Switch 액티브 아래에 추가합니다.



추가한 Invoke 액티브와 FufillmentProcess(BPEL Process) 컴포넌트와 연결합니다.

Name : Invoke_Fulfillment

Input Variable을 기본 이름으로 추가합니다.



Transform 액티브를 Invoke_Fulfillment 액티브 위에 추가합니다.



추가한 Transform 액티브를 더블 클릭하여 편집합니다.



Source Variable 은 inputVariablepayload를 선택합니다.



Target Variable은 Invoke_Fulfillment_process_InputVariablepayload를 선택합니다.

Create Mapping 버튼을 클릭합니다.



customerIdorderId를 매핑합니다.



Mathematical Functions - multiply 을 가운데 영역에 추가합니다.

그리고 priceqty를 추가한 multiply과 연결하고, totalmultiply과 연결합니다.



Switch 액티브를 Invoke_Fulfillment 액티브 바로 액티브 아래에 추가합니다.



Sequence 액티브를 Switch 액티브의 <case> 영역 안에 추가합니다.



Transform_1을 추가한 Sequence 안으로 옮깁니다.



Invoke_Fulfillment를 Sequence안 Transform_1 아래로 옮깁니다.



<case>의 조건을 아래와 같이 설정합니다. (XPath Expression Builder 를 이용하세요)



<other wish> 부분은 삭제합니다.





routePO를 더블 클릭하여 수정합니다.



static routing rule을 추가합니다.



Service 클릭



타겟 서비스를 선택합니다.



맵핑을 선택합니다.



맵핑파일을 생성합니다.



아래와 같이 맵핑 합니다.



Mediator 설정에서 FulfillmentProcess로 가는 조건을 추가합니다.





모두 저장합니다~ 완성되었습니다.





Deploying and testing the application

테스트를 해보면 배송 처리가 된 것을 확인할 수 있습니다.







반응형
//