Thursday, 6 March 2025

Oracle advance queue for configuration on oracle 19c for ANYDATA type.

Configuring Oracle Data Queue for ANYDATA Payload

In this guide, we will walk through the process of creating and configuring an Oracle data queue for an ANYDATA payload. This setup allows seamless data transfer between two databases (DB1 and DB2) while ensuring data integrity and type flexibility.

Prerequisites

  • Two databases (DB1 and DB2) with connectivity.

  • The global_name parameter must be enabled on both databases.

  • A database link must be created between DB1 and DB2.


Step 1: Enable Global Name Parameter

Run the following command as SYSDBA to check the status of the global_name parameter:

SHOW PARAMETER global_name;

If the value is FALSE, set it to TRUE using:

ALTER SYSTEM SET global_name = TRUE;

Step 2: Create a Database Link

Create a database link from DB1 to DB2 for communication:

CREATE DATABASE LINK "DB2"
CONNECT TO "STRMADMIN"
IDENTIFIED BY "strmadmin"
USING '(DESCRIPTION =(ADDRESS = (PROTOCOL = TCP)(HOST = 192.168.1.1)(PORT = 1521))(CONNECT_DATA = (SID=ORCL)))';

Test the connection:

SELECT * FROM global_name@DB2;

If successful, this should return the global_name of the destination database (DB2).


Step 3: Create a User-Defined Type

The following user-defined type must be created on both databases:

CREATE OR REPLACE EDITIONABLE TYPE STRMADMIN.MYTYPE AS OBJECT (
    ID NUMBER,
    REMARK VARCHAR2(100 BYTE)
);
/

Step 4: Configure the Queue System

Drop Existing Queues (If Any)

Before creating a new queue, ensure that any existing queues are stopped and removed:

BEGIN
    DBMS_AQADM.STOP_QUEUE(queue_name => 'strmadmin.test_q');
    DBMS_AQADM.DROP_QUEUE(queue_name => 'strmadmin.test_q');
    DBMS_AQADM.DROP_QUEUE_TABLE(queue_table => 'strmadmin.test_qt');
END;
/

If the above commands do not work, remove the queue using:

BEGIN
    DBMS_STREAMS_ADM.REMOVE_QUEUE(
        queue_name               =>  'strmadmin.test_q',
        cascade                  =>  TRUE,
        drop_unused_queue_table  =>  TRUE);
END;
/

Create a Queue on DB1

BEGIN
    DBMS_AQADM.create_queue_table(
        queue_table => 'test_qt',
        queue_payload_type => 'SYS.ANYDATA',
        multiple_consumers => TRUE
    );

    DBMS_AQADM.create_queue(
        queue_name => 'test_q',
        queue_table => 'test_qt'
    );

    DBMS_AQADM.start_queue(queue_name => 'test_q');
END;
/

Create Propagation on DB1

Propagation should be created after the queue is established on both sides.

BEGIN
    DBMS_PROPAGATION_ADM.CREATE_PROPAGATION(
        propagation_name   => 'TEST_PROP',
        source_queue       => 'strmadmin.test_q',
        destination_queue  => 'strmadmin.test_q',
        destination_dblink => 'DB2',
        queue_to_queue     => TRUE
    );
END;
/

Create a Queue on DB2 (for Receiving Messages)

BEGIN
    DBMS_AQADM.create_queue_table(
        queue_table => 'test_qt',
        queue_payload_type => 'SYS.ANYDATA',
        multiple_consumers => TRUE
    );

    DBMS_AQADM.create_queue(
        queue_name => 'test_q',
        queue_table => 'test_qt'
    );

    DBMS_AQADM.start_queue(queue_name => 'test_q');
    
    DBMS_AQADM.add_subscriber(
        queue_name => 'test_q',
        subscriber => SYS.AQ$_AGENT('RECIPIENT', NULL, NULL)
    );
END;
/

Step 5: Testing the Queue System

Enqueue a Message from DB1

declare
    enqueue_options    DBMS_AQ.ENQUEUE_OPTIONS_T;
    message_properties DBMS_AQ.MESSAGE_PROPERTIES_T;
    message_handle     RAW(16);
    MESSAGE            STRMADMIN.MYTYPE;
    msg                ANYDATA;
begin
    MESSAGE := STRMADMIN.MYTYPE(1, 'TEST');
    msg := ANYDATA.CONVERTOBJECT(MESSAGE);

    DBMS_AQ.ENQUEUE(
        queue_name         => 'STRMADMIN.test_q',
        enqueue_options    => enqueue_options,
        message_properties => message_properties,
        payload            => msg,
        msgid              => message_handle
    );
    COMMIT;
end;
/

Dequeue Manually from DB2

DECLARE
    dequeue_options     DBMS_AQ.dequeue_options_t;
    message_properties  DBMS_AQ.message_properties_t;
    msgid               RAW(16);
    MESSAGE             STRMADMIN.MYTYPE;
    msg                 ANYDATA;
BEGIN
    FOR rec IN (SELECT * FROM test_qt)
LOOP dequeue_options.msgid := rec.msgid; dequeue_options.consumer_name := 'RECIPIENT'; DBMS_AQ.DEQUEUE( queue_name => 'STRMADMIN.TEST_Q', dequeue_options => dequeue_options, message_properties => message_properties, payload => msg, msgid => msgid ); IF msg.GETTYPENAME() = 'STRMADMIN.MYTYPE' THEN rc := msg.GETOBJECT(MESSAGE); DBMS_OUTPUT.PUT_LINE('ID: ' || MESSAGE.ID); DBMS_OUTPUT.PUT_LINE('REMARK: ' || MESSAGE.REMARK); END IF; END LOOP; COMMIT; EXCEPTION WHEN OTHERS THEN ROLLBACK; END; /

Step 6: Register a Callback Procedure

If you want automatic message dequeuing upon arrival, register a callback procedure:

CREATE OR REPLACE PROCEDURE DEQUEUE_MSG_CB(
  context  IN RAW,
  reginfo  IN SYS.AQ$_REG_INFO, 
  descr    IN SYS.AQ$_DESCRIPTOR,
  payload  IN VARCHAR2,
  payloadl IN NUMBER
)
AS
   MESSAGE             STRMADMIN.MYTYPE;
   msg                 ANYDATA;
BEGIN
    -- Get the message identifier and consumer name from the descriptor
	dequeue_options.msgid := descr.msg_id;
	dequeue_options.consumer_name := descr.consumer_name;
    -- Dequeue the message
	DBMS_AQ.DEQUEUE(
		queue_name         => descr.queue_name,
		dequeue_options    => dequeue_options,
		message_properties => message_properties,
		payload            => msg,
		msgid              => message_handle);

	if msg.GETTYPENAME() = 'STRMADMIN.MYTYPE' then
		rc := msg.getobject(MESSAGE);
		DBMS_OUTPUT.PUT_LINE('ID ' || MESSAGE.ID);
		DBMS_OUTPUT.PUT_LINE('REMARK '|| MESSAGE.REMARK);
	end if;

  -- if you are saving data
  COMMIT;
END;
/

Register the callback procedure:

BEGIN  
    DBMS_AQ.REGISTER(
        SYS.AQ$_REG_INFO_LIST(
            SYS.AQ$_REG_INFO(
                'test_q:RECIPIENT',
                DBMS_AQ.NAMESPACE_AQ,
                'plsql://strmadmin.DEQUEUE_MSG_CB?PR=1',
                HEXTORAW('FF')
            )
        ), 1
    );
END;
/

This completes the setup of an Oracle ANYDATA queue system, ensuring reliable data propagation between databases!

References

For more details, refer to the Oracle Advanced Queuing Guide.

Tuesday, 13 June 2023

RegEx in EXCEL

' @info: function find server name "xx00x00x" from give string at the begining of string or srever name begin with space
' @return: string | name of server
Public Function getServer(str As String) As String
On Error GoTo Err_getServer
    Dim regex As Object
    Set regex = CreateObject("VBScript.RegExp")
     
    With regex
        .Pattern = "^(\s*)[A-Z][A-Z][0-9][0-9][A-Z][0-9][0-9][A-Z]*|\s[A-Z][A-Z][0-9][0-9][A-Z][0-9][0-9][A-Z]*"
        .IgnoreCase = True
        .Global = True
    End With
 
    Set matches = regex.Execute(str)
     
    For Each Match In matches
        Debug.Print Match.Value '
        getServer = getServer + Match.Value
    Next Match
     
    'remove space
    getServer = Replace(getServer, " ", "", 1, 1)
     
Ext_getServer:
    Exit Function
 
Err_getServer:
   getServer = CVErr(xlErrValue)
   GoTo Ext_getServer
End Function


' @info: function find server name "xx00x00x" from give string at the begining of string or srever name begin with space in string and replace with ""
' @return: string | clear string with out server name
Public Function getClearStr(str As String) As String
On Error GoTo Err_getClearStr
    Dim regex As Object
    Dim temp As String
    Set regex = CreateObject("VBScript.RegExp")
       
    With regex
        .Pattern = "^(\s*)[A-Z][A-Z][0-9][0-9][A-Z][0-9][0-9][A-Z]*|\s[A-Z][A-Z][0-9][0-9][A-Z][0-9][0-9][A-Z]*"
        .IgnoreCase = True
        .Global = True
    End With
   
    Set matches = regex.Execute(str)
       
    For Each Match In matches
        Debug.Print Match.Value '
        temp = Match.Value
    Next Match
   
    getClearStr = Replace(str, temp, "", 1, 1)
   
    If (Left(getClearStr, 1) = ";") Or (Left(getClearStr, 1) = ",") Then
        If Len(getClearStr) > 1 Then
            getClearStr = Right(getClearStr, Len(getClearStr) - 1)
        End If
    End If
   
Exit_getClearStr:
    Exit Function
   
Err_getClearStr:
    getClearStr = CVErr(xlErrValue)
    GoTo Exit_getClearStr
End Function

Oracle Apex | get text message into sql and javaScript when implement translation

Translate text message

When you want to translate text into plsql code:

 

Then you have to define text with key in different language in shared components > translate > Text Message

 

And retrieve message through

APEX_LANG.MESSAGE('MSG_AVAILABLE');

 

 

When want to translated text into javaScript code:

 

Then you have to define text like below in different language in shared components > translate > Text Message

NOTE:= set "Used in javaScript" to "Yes"

 


Then you can retrieve message through following code:

apex.lang.getMessage('MSG_ERROR_CASE_SEARCH');

Thursday, 6 July 2017

Oracle Procedure to send html email with multiple attachment [ blob]

create or replace TYPE attach_info IS OBJECT (
        value         VARCHAR2(100),
        mimetype      VARCHAR2(100),
        content       BLOB
    );
/

create or replace TYPE array_attachments IS TABLE OF attach_info;
/

PROCEDURE HTML_EMAIL_MUlTI_BLOBS (
/* TO SEND HTML OR PLAN TEXT EMAIL WITH ATTACHMENTS (BLOB)
-- TWO TYPES SHOULD BE IN DATABASE FOR THIS PROCEDURE WHICH ARE FOLLOWING:
--   create or replace TYPE attach_info IS OBJECT (
--        value         VARCHAR2(100),
--        mimetype      VARCHAR2(100),
--        content       BLOB
--    );
-- create or replace TYPE array_attachments IS TABLE OF attach_info;
*/
                                       p_to          IN VARCHAR2,
                                       p_cc          IN VARCHAR2 DEFAULT NULL,
                                       p_from        IN VARCHAR2,
                                       p_subject     IN VARCHAR2,
                                       p_text_msg    IN VARCHAR2 DEFAULT NULL,
                                       p_html_msg    IN VARCHAR2 DEFAULT NULL,
                                       attachments   IN OUT array_attachments,
                                       p_smtp_host   IN VARCHAR2,
                                       p_smtp_port   IN NUMBER DEFAULT 25)
AS
  l_mail_conn   UTL_SMTP.connection;
  l_boundary    VARCHAR2(50) := '----=*#abc1234321cba#*=';
  l_step        PLS_INTEGER  := 12000; -- make sure you set a multiple of 3 not higher than 24573
BEGIN
  l_mail_conn := UTL_SMTP.open_connection(p_smtp_host, p_smtp_port);
  UTL_SMTP.helo(l_mail_conn, p_smtp_host);
  UTL_SMTP.mail(l_mail_conn, p_from);
  UTL_SMTP.rcpt(l_mail_conn, p_to);
  IF TRIM(p_cc) IS NOT NULL THEN
    UTL_SMTP.rcpt(l_mail_conn, p_cc);
  END IF;
  UTL_SMTP.open_data(l_mail_conn);

  UTL_SMTP.write_data(l_mail_conn, 'Date: ' || TO_CHAR(SYSDATE, 'DD-MON-YYYY HH24:MI:SS') || UTL_TCP.crlf);
  UTL_SMTP.write_data(l_mail_conn, 'To: ' || p_to || UTL_TCP.crlf);
  IF TRIM(p_cc) IS NOT NULL THEN    
      UTL_SMTP.write_raw_data(l_mail_conn, utl_raw.cast_to_raw('CC: ' || REPLACE(p_cc, ',', ';') || UTL_TCP.crlf));
  END IF;
  UTL_SMTP.write_data(l_mail_conn, 'From: ' || p_from || UTL_TCP.crlf);
  UTL_SMTP.write_data(l_mail_conn, 'Subject: ' || p_subject || UTL_TCP.crlf);
  UTL_SMTP.write_data(l_mail_conn, 'Reply-To: ' || p_from || UTL_TCP.crlf);
  UTL_SMTP.write_data(l_mail_conn, 'MIME-Version: 1.0' || UTL_TCP.crlf);
  UTL_SMTP.write_data(l_mail_conn, 'Content-Type: multipart/mixed; boundary="' || l_boundary || '"' || UTL_TCP.crlf || UTL_TCP.crlf);

  -- check plain text message and set
  IF p_text_msg IS NOT NULL THEN
    UTL_SMTP.write_data(l_mail_conn, '--' || l_boundary || UTL_TCP.crlf);
    UTL_SMTP.write_data(l_mail_conn, 'Content-Type: text/plain; charset="iso-8859-1"' || UTL_TCP.crlf || UTL_TCP.crlf);

    UTL_SMTP.write_data(l_mail_conn, p_text_msg);
    UTL_SMTP.write_data(l_mail_conn, UTL_TCP.crlf || UTL_TCP.crlf);
  END IF;

 -- check html message and set
 IF p_html_msg IS NOT NULL THEN
    UTL_SMTP.write_data(l_mail_conn, '--' || l_boundary || UTL_TCP.crlf);
    UTL_SMTP.write_data(l_mail_conn, 'Content-Type: text/html; charset="iso-8859-1"' || UTL_TCP.crlf || UTL_TCP.crlf);

    UTL_SMTP.write_data(l_mail_conn, p_html_msg);
    UTL_SMTP.write_data(l_mail_conn, UTL_TCP.crlf || UTL_TCP.crlf);
  END IF;

  -- Attachment Part
  IF attachments IS NOT NULL THEN
    FOR i IN attachments.FIRST .. attachments.LAST
    LOOP
    -- Attach info
        UTL_SMTP.write_raw_data(l_mail_conn, utl_raw.cast_to_raw('--' || l_boundary || UTL_TCP.crlf));
        UTL_SMTP.write_raw_data(l_mail_conn, utl_raw.cast_to_raw('Content-Type: ' || attachments(i).mimetype
                            || ' name="'|| attachments(i).value || '"' || UTL_TCP.crlf));
        UTL_SMTP.write_data(l_mail_conn, 'Content-Transfer-Encoding: base64' || UTL_TCP.crlf);
        UTL_SMTP.write_raw_data(l_mail_conn, utl_raw.cast_to_raw('Content-Disposition: attachment; filename="'
                            || attachments(i).value || '"' || UTL_TCP.crlf || UTL_TCP.crlf));

    -- Attach body
      FOR j IN 0 .. TRUNC((DBMS_LOB.getlength(attachments(i).content) - 1 )/l_step) LOOP
        UTL_SMTP.write_data(l_mail_conn, UTL_RAW.cast_to_varchar2(UTL_ENCODE.base64_encode(DBMS_LOB.substr(attachments(i).content, l_step, j * l_step + 1))));
      END LOOP;

         UTL_SMTP.write_data(l_mail_conn, UTL_TCP.crlf || UTL_TCP.crlf);
    END LOOP;
  END IF;

  UTL_SMTP.write_data(l_mail_conn, '--' || l_boundary || '--' || UTL_TCP.crlf);
  UTL_SMTP.close_data(l_mail_conn);

  UTL_SMTP.quit(l_mail_conn);
END;

/


How to initializing attachments:

declare:
attachments         array_attachments;

assign:
attachments := array_attachments();



attachments.extend(2);

attachments(1):= attach_info(null,null,null); /*extend it and initialize attach_info object and only then assign attachments(1).value, attachments(1).mimetype, attachments(1).content https://community.oracle.com/message/10085562*/


      SELECT oa0.VALUE||'.'||oa19.value,
            mty.mimetype,
            o.CONTENT
        INTO attachments(1).value,
             attachments(1).mimetype,
             attachments(1).content
        FROM sdms.OBJECTS o
        JOIN sdms.OBJECTATTRIBUTES oa0 ON (o.ID = oa0.OBJECT_ID AND oa0.ATTRIBUTEDESCRIPTION_ID = 0) -- FileName
        JOIN sdms.OBJECTATTRIBUTES oa19 ON (o.ID = oa19.OBJECT_ID AND oa19.ATTRIBUTEDESCRIPTION_ID = 19), -- FileExtension
        sdms.t_mime_types mty
        WHERE o.mime_id=mty.id AND o.id=9124;


Calling procedure:

HTML_EMAIL_MUlTI_BLOBS(v_to, v_cc, v_from, v_subject, null, cMessage, attachments, v_smtp_host);