Oracle과 MS-SQL 사이의 Data 공유를 위해서 Openquery를 이용하고 있습니다. 지금까지 잘 이용하고 있었는데, 문제가 발생했습니다. Oracle DB를 Tibero로 바꾼다는거죠. 그래서 Tibero에서도 해당 기능을 사용할 수 있는지 점검해 봤습니다.
Tibero to MSSQL Gateway라는게 존재하는데, 이는 설정을 따로 요청해야 하고, 기존의 MS-SQL측에서 생성된 프로시져를 쓸 수가 없다는 단점이 있어서 일단 Openquery를 이용하는 방법을 시도해봤습니다.
MS-SQL에서 Linked server를 생성하는 방법에는 Tibero ODBC driver를 이용해서 Data 원본(DSN)을 만든 뒤에 이를 이용해서 Linked server를 생성하는 법과, Tibero oledb driver(2가지를 지원하더군요.)를 이용해서 바로 Linked server를 생성하는 방법이 있습니다.
1. select query 실행
ODBC, OLEDB 두가지 방법으로 모두 이상없이 잘 됩니다.
2. update query 실행
ODBC, OLEDB 두가지 방법으로 모두 오류가 발생합니다.
결론.
Openquery를 이용할 때 Tibero로는 select query만 실행 가능하다는 문제가 있습니다.
MS-SQL에서 Oracle에 있는 Data에 접근하기 위한 방법으로 Openquery라는걸 소개한 적이 있죠.
기본적으로 제공하는 기능이라 좋긴 하지만 동적으로 쿼리를 만들어서 결과값을 받아올수 없다는 얘기를 한 적이 있습니다. 그렇게 되면 WHERE절 조건을 정확하게 줄 수 없으니 오라클 DB에 부하를 많이 주게 되겠죠.
저도 이렇게 알고 서비스 중인 MS-SQL과 Oracle 사이에 Data 동기화 프로시져를 만들었었습니다.
그런데....
웬걸...
회사에서 사용중인 MS-SQL 2005에서 혹시나 하는 마음에 문자열 변수에 커서를 정의하는 문장까지 포함해서 동적으로 쿼리를 만들어 주고, 커서를 열었더니... 결과 값이 정상적으로 나오네요. 앗싸~ 가오리~
아래와 같은 방식으로 처리하니까 동적으로 쿼리문장을 만들어서 실행시키고 결과값을 커서에 받아서 사용할 수 있습니다.
SET @QUERY_STRING = 'DECLARE ORA_CUR CURSOR FOR
SELECT USERNAME
FROM OPENQUERY(ORA_DB, ''SELECT USERNAME
FROM USER_INFO
WHERE CODE='''''+@CODE+''''' AND USER_FLAG=''''N'''' '') '
EXEC (@QUERY_STRING)
OPEN ORA_CUR
FETCH NEXT FROM ORA_CUR INTO @USERNAME
WHILE @@FETCH_STATUS = 0
BEGIN
INSERT INTO dbo.USER_INFO(USERNAME)
VALUES(@USERNAME)
새로 도입하는 시스템이 MS-SQL을 사용한다고 해서 기존에 사용하던 Oracle에서 일부 Data는 초기에 한번, 일부 Data는 주기적으로 동기화를 맞춰주는 작업을 해야합니다. 초기에 한번 이전해 주는 작업은 MS-SQL의 SSIS라는 기능을 이용해서 넘겨주기로 했고, 기능 테스트도 이미 했죠. 문제는 주기적으로 Data의 동기화를 맞춰주는 작업인데요. DB 복제 솔루션을 이용하거나 ETL을 이용하면 쉽게 해결되겠지만, 예산이 없어서 보류하고요. 다른 방법을 찾아보기로 결정했었죠. 그래서 알아본게 다음의 두가지입니다.
1. Oracle Gateway를 이용한 이기종 DBMS간 Data 공유
2. MS-SQL의 Open Query를 이용한 이기종 DBMS간 Data 공유
Oracle DB에 원천 Data가 있으니 변경된 내용이 생길때 바로바로 적용하기 위해서는 Oracle Gateway를 이용하는게 좋을것 같긴 한데요. 문제는 이건 제가 한번도 해보질 않아서 프로젝트 일정에 맞출수가 없을 것 같더군요. 그래서 MS-SQL에서 구현하기로 했습니다. Open query는 예전에도 써본적이 있거든요.
자꾸 이러면 실력이 늘지 않을텐데 큰일입니다. ^^; 자~ 이제 MS-SQL의 Open query를 이용해서 주기적으로 Data 동기화를 맞춰주는 기능을 구현해보도록 하겠습니다.
제일 먼저 할 일은 MS-SQL 서버에 Oracle client를 설치하는 일입니다. 설치가 완료되면 Data를 가져올 Oracle server의 정보를 tnsnames.ora에 설정해주면 됩니다. 이제 Linked server를 설정해줄 차례입니다.
Open Query를 사용하기 위해서 우선 Oracle과 Link를 걸어야 합니다. SQL Server Management Studio에서 [연결된 서버]라는 항목에 추가를 해주면 됩니다. 이때 [Oracle provider for OLEDB]를 선택해서 사용하시는게 좋습니다. MS에서 제공하는 OLEDB provider를 사용할 수도 있는데, 이건 뭔가 문제가 있어보이더군요. Open query에서는 테스트해보지 않았지만, SSIS에서 Oracle의 Data를 가져올때 데이터형을 좀 잘못 가져오는것 같더군요. tnsnames.ora에 설정된 이름과 사용할 Oracle의 계정 정보 등을 입력해서 linked server를 생성하시면 됩니다.
이제 Open query에 대해서 얘기해볼까합니다. Open query를 이용하면 MS-SQL에서 Oracle DB에 insert, select, update, delete를 실행 할 수 있습니다.
USE TESTDB
GO
SELECT * FROM OPENQUERY(LINK_NAME, `SELECT * FROM 테이블이름`);
UPDATE OPENQUERY(LINK_NAME, `SELECT A FROM 테이블이름 WHERE 조건1`)
SET A='AA' WHERE 조건2;
DELETE OPENQUERY(LINK_NAME, `SELECT A FROM 테이블이름 WHERE 조건1`)
위의 query문을 보시면 update문을 실행시키는데 select문을 실행시키고 set을 해주는 부분이 보일겁니다. 그리고 조건이 2개 붙은 부분이 보일겁니다. [조건2]가 붙는 이유는 [조건1]에 ms-sql의 프로시져에서 사용하는 변수를 적용할 수 없기 때문입니다. 변수로 처리하지 않고 문자열로 넣어서 처리할 수 있는 부분은 [조건1]에 다 넣어주고, 변수로 받아서 넣어줘야할 내용은 [조건2]에 넣어주면 됩니다. 아래에 프로시져 코드를 보시면 바로 이해하실 겁니다.
이제 배치작업을 위해서 해당 기능을 프로시져로 만듭니다. 제가 작성한 간단한 샘플 코드를 보여드리겠습니다.
USE [testdb]
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
CREATE PROCEDURE [tdb].[user_sync]
AS
DECLARE ora_cursor CURSOR FOR
SELECT TRG_FLAG, USER_ID, USER_NM, TEL, CEL, EMAIL, ZIP_CD, ADDR1, ADDR2
FROM OPENQUERY(VOCA2, 'SELECT B.TRG_FLAG, B.USER_ID, A.USER_NM, A.TEL, A.CEL,
A.EMAIL, A.ZIP_CD, A.ADDR1, A.ADDR2
FROM USER_TEST A, USER_TRG B
WHERE B.USER_ID = A.USER_ID(+) AND B.MIG_DATE IS NULL');
FETCH NEXT FROM ora_cursor INTO @TRG_FLAG, @UM_ID, @UM_NM, @UM_TEL, @UM_CEL, @UM_EMAIL, @ZIP_CD, @UM_ADDR1, @UM_ADDR2, @JOB_CD, @UM_FLAG, @REG_STRT_DAY, @LS_E_USER, @UM_EN_NM, @USE_YN, @UM_NIC_NM
WHILE @@FETCH_STATUS = 0
BEGIN
IF (@TRG_FLAG = 'I')
BEGIN
INSERT INTO GNB.GE_USR_TEST(UM_ID, UM_NM, UM_TEL, UM_CEL, UM_EMAIL, ZIP_CD, UM_ADDR1, UM_ADDR2, JOB_CD, UM_FLAG,
REG_STRT_DAY, LS_E_USER, UM_EN_NM, USE_YN, UM_NIC_NM)
VALUES(@UM_ID, @UM_NM, @UM_TEL, @UM_CEL, @UM_EMAIL, @ZIP_CD, @UM_ADDR1, @UM_ADDR2, @JOB_CD, @UM_FLAG,
@REG_STRT_DAY, @LS_E_USER, @UM_EN_NM, @USE_YN, @UM_NIC_NM);
UPDATE OPENQUERY(VOCA2, 'SELECT MIG_FLAG, MIG_DATE, UM_ID FROM GE_USR_TRG WHERE TRG_FLAG=''I'' AND MIG_FLAG=''N''')
SET MIG_FLAG='Y', MIG_DATE=GETDATE()
WHERE UM_ID=@UM_ID;
END;
ELSE IF (@TRG_FLAG = 'U')
BEGIN
UPDATE GNB.GE_USR_TEST SET UM_NM=@UM_NM, UM_TEL=@UM_TEL, UM_CEL=@UM_CEL, UM_EMAIL=@UM_EMAIL, ZIP_CD=@ZIP_CD,
UM_ADDR1=@UM_ADDR1, UM_ADDR2=@UM_ADDR2, JOB_CD=@JOB_CD, UM_FLAG=@UM_FLAG, REG_STRT_DAY=@REG_STRT_DAY,
LS_E_USER=@LS_E_USER, UM_EN_NM=@UM_EN_NM, USE_YN=@USE_YN, UM_NIC_NM=@UM_NIC_NM
WHERE UM_ID=@UM_ID;
UPDATE OPENQUERY(VOCA2, 'SELECT MIG_FLAG, MIG_DATE, UM_ID FROM GE_USR_TRG WHERE TRG_FLAG=''U'' AND MIG_FLAG=''N''')
SET MIG_FLAG='Y', MIG_DATE=GETDATE()
WHERE UM_ID=@UM_ID;
END;
ELSE IF (@TRG_FLAG = 'D')
BEGIN
DELETE FROM GNB.GE_USR_TEST WHERE UM_ID=@UM_ID;
UPDATE OPENQUERY(VOCA2, 'SELECT MIG_FLAG, MIG_DATE, UM_ID FROM GE_USR_TRG WHERE TRG_FLAG=''D'' AND MIG_FLAG=''N''')
SET MIG_FLAG='Y', MIG_DATE=GETDATE()
WHERE UM_ID=@UM_ID;
END;
INSERT INTO DBO.GE_USR_TEST_LOG(REG_DATE, UM_ID, TRG_FLAG) VALUES(GETDATE(), @UM_ID, @TRG_FLAG);
FETCH NEXT FROM ora_cursor INTO @TRG_FLAG, @UM_ID, @UM_NM, @UM_TEL, @UM_CEL, @UM_EMAIL, @ZIP_CD, @UM_ADDR1, @UM_ADDR2, @JOB_CD, @UM_FLAG, @REG_STRT_DAY, @LS_E_USER, @UM_EN_NM, @USE_YN, @UM_NIC_NM
END;