zoukankan      html  css  js  c++  java
  • Configuring Service Broker for Asynchronous Processing

    Configuring Service Broker for Asynchronous Processing

    --create a database and enable the database for Service Broker usage
    
    CREATE DATABASE AsyncProcessingDemo;
    GO
     
    IF (SELECT is_broker_enabled FROM sys.databases WHERE name = N'AsyncProcessingDemo') = 0
    BEGIN
      ALTER DATABASE AsyncProcessingDemo SET ENABLE_BROKER;
    END
    GO
     
    USE AsyncProcessingDemo;
    GO
    
    --=========================================================================================================
    --Configuring broker components
    -- Create the message types
    CREATE MESSAGE TYPE [AsyncRequest] VALIDATION = WELL_FORMED_XML;
    CREATE MESSAGE TYPE [AsyncResult]  VALIDATION = WELL_FORMED_XML;
    
    -- Create the contract
    CREATE CONTRACT [AsyncContract] 
    (
      [AsyncRequest] SENT BY INITIATOR, 
      [AsyncResult]  SENT BY TARGET
    );
    
    -- Create the processing queue and service - specify the contract to allow sending to the service
    CREATE QUEUE ProcessingQueue;
    CREATE SERVICE [ProcessingService] ON QUEUE ProcessingQueue ([AsyncContract]);
     
    -- Create the request queue and service 
    CREATE QUEUE RequestQueue;
    CREATE SERVICE [RequestService] ON QUEUE RequestQueue;
    
    --=========================================================================================================
    --Sending a Message for Processing
    -- Create the wrapper procedure for sending messages
    CREATE PROCEDURE dbo.SendBrokerMessage 
        @FromService SYSNAME,
        @ToService   SYSNAME,
        @Contract    SYSNAME,
        @MessageType SYSNAME,
        @MessageBody XML
    AS
    BEGIN
      SET NOCOUNT ON;
     
      DECLARE @conversation_handle UNIQUEIDENTIFIER;
     
      BEGIN TRANSACTION;
     
      BEGIN DIALOG CONVERSATION @conversation_handle
        FROM SERVICE @FromService
        TO SERVICE @ToService
        ON CONTRACT @Contract
        WITH ENCRYPTION = OFF;
     
      SEND ON CONVERSATION @conversation_handle
        MESSAGE TYPE @MessageType(@MessageBody);
     
      COMMIT TRANSACTION;
    END
    GO
    
    -- Send a request
    EXECUTE dbo.SendBrokerMessage
      @FromService = N'RequestService',
      @ToService   = N'ProcessingService',
      @Contract    = N'AsyncContract',
      @MessageType = N'AsyncRequest',
      @MessageBody = N'<AsyncRequest><AccountNumber>12345</AccountNumber></AsyncRequest>';
     
    -- Check for message on processing queue
    SELECT CAST(message_body AS XML) FROM ProcessingQueue;
    GO
    
    --=========================================================================================================
    --Processing Messages
    -- Create processing procedure for processing queue
    CREATE PROCEDURE dbo.ProcessingQueueActivation
    AS
    BEGIN
      SET NOCOUNT ON;
     
      DECLARE @conversation_handle UNIQUEIDENTIFIER;
      DECLARE @message_body XML;
      DECLARE @message_type_name sysname;
     
      WHILE (1=1)
      BEGIN
        BEGIN TRANSACTION;
     
        WAITFOR
        (
          RECEIVE TOP (1)
            @conversation_handle = conversation_handle,
            @message_body = CAST(message_body AS XML),
            @message_type_name = message_type_name
          FROM ProcessingQueue
        ), TIMEOUT 5000;
     
        IF (@@ROWCOUNT = 0)
        BEGIN
          ROLLBACK TRANSACTION;
          BREAK;
        END
     
        IF @message_type_name = N'AsyncRequest'
        BEGIN
          -- Handle complex long processing here
          -- For demonstration we'll pull the account number and send a reply back only
     
          DECLARE @AccountNumber INT = @message_body.value('(AsyncRequest/AccountNumber)[1]', 'INT');
     
          -- Build reply message and send back
          DECLARE @reply_message_body XML = N'
            ' + CAST(@AccountNumber AS NVARCHAR(11)) + '
          ';
     
          SEND ON CONVERSATION @conversation_handle
            MESSAGE TYPE [AsyncResult] (@reply_message_body);
        END
     
        -- If end dialog message, end the dialog
        ELSE IF @message_type_name = N'http://schemas.microsoft.com/SQL/ServiceBroker/EndDialog'
        BEGIN
          END CONVERSATION @conversation_handle;
        END
     
        -- If error message, log and end conversation
        ELSE IF @message_type_name = N'http://schemas.microsoft.com/SQL/ServiceBroker/Error'
        BEGIN
          -- Log the error code and perform any required handling here
          -- End the conversation for the error
          END CONVERSATION @conversation_handle;
        END
     
        COMMIT TRANSACTION;
      END
    END
    GO
    
    -- Create procedure for processing replies to the request queue
    CREATE PROCEDURE dbo.RequestQueueActivation
    AS
    BEGIN
      SET NOCOUNT ON;
     
      DECLARE @conversation_handle UNIQUEIDENTIFIER;
      DECLARE @message_body XML;
      DECLARE @message_type_name sysname;
     
      WHILE (1=1)
      BEGIN
        BEGIN TRANSACTION;
     
        WAITFOR
        (
          RECEIVE TOP (1)
            @conversation_handle = conversation_handle,
            @message_body = CAST(message_body AS XML),
            @message_type_name = message_type_name
          FROM RequestQueue
        ), TIMEOUT 5000;
     
        IF (@@ROWCOUNT = 0)
        BEGIN
          ROLLBACK TRANSACTION;
          BREAK;
        END
     
        IF @message_type_name = N'AsyncResult'
        BEGIN
          -- If necessary handle the reply message here
          DECLARE @AccountNumber INT = @message_body.value('(AsyncResult/AccountNumber)[1]', 'INT');
     
          -- Since this is all the work being done, end the conversation to send the EndDialog message
          END CONVERSATION @conversation_handle;
        END
     
        -- If end dialog message, end the dialog
        ELSE IF @message_type_name = N'http://schemas.microsoft.com/SQL/ServiceBroker/EndDialog'
        BEGIN
           END CONVERSATION @conversation_handle;
        END
     
        -- If error message, log and end conversation
        ELSE IF @message_type_name = N'http://schemas.microsoft.com/SQL/ServiceBroker/Error'
        BEGIN
           END CONVERSATION @conversation_handle;
        END
     
        COMMIT TRANSACTION;
      END
    END
    GO
    
    --=========================================================================================================
    --Testing the Procedures
    -- Process the message from the processing queue
    EXECUTE dbo.ProcessingQueueActivation;
    GO
     
    -- Check for reply message on request queue
    SELECT CAST(message_body AS XML) FROM RequestQueue;
    GO
     
    -- Process the message from the request queue
    EXECUTE dbo.RequestQueueActivation;
    GO
    
    --=========================================================================================================
    --Automating the Processing
    -- Alter the processing queue to specify internal activation
    ALTER QUEUE ProcessingQueue
        WITH ACTIVATION
        ( 
          STATUS = ON,
          PROCEDURE_NAME = dbo.ProcessingQueueActivation,
          MAX_QUEUE_READERS = 10,
          EXECUTE AS SELF
        );
    GO
     
    -- Alter the request queue to specify internal activation
    ALTER QUEUE RequestQueue
        WITH ACTIVATION
        ( 
          STATUS = ON,
          PROCEDURE_NAME = dbo.RequestQueueActivation,
          MAX_QUEUE_READERS = 10,
          EXECUTE AS SELF
        );
    GO
     
    -- Test automated activation
    -- Send a request
     
    EXECUTE dbo.SendBrokerMessage
        @FromService = N'RequestService',
        @ToService   = N'ProcessingService',
        @Contract    = N'AsyncContract',
        @MessageType = N'AsyncRequest',
        @MessageBody = N'<AsyncRequest><AccountNumber>12345</AccountNumber></AsyncRequest>';
     
    -- Check for message on processing queue 
    -- nothing is there because it was automatically processed
    SELECT CAST(message_body AS XML) FROM ProcessingQueue;
    GO
     
    -- Check for reply message on request queue 
    -- nothing is there because it was automatically processed
    SELECT CAST(message_body AS XML) FROM RequestQueue;
    GO

    转自:https://sqlperformance.com/2014/03/sql-performance/configuring-service-broker

  • 相关阅读:
    Django环境配置
    python pip使用报错: Fatal error in launcher: Unable to create process using '"c:python27python.exe" "C:Python27Scriptspip.exe" '
    我的第100篇博客
    测试代码(测试函数)
    python 存储数据
    python 异常之进阶操作
    从实战的角度谈微服务(三):基于Feign的熔断
    从实战的角度谈微服务(三):基于Feign服务间的调用
    从实战的角度谈微服务(二):服务注册与发现(Eureka服务端)
    从实战的角度谈微服务(一):初识微服务
  • 原文地址:https://www.cnblogs.com/davidhou/p/5916680.html
Copyright © 2011-2022 走看看