I've got a very occasional problem whereby a process is blocking the incremental rebuild of some fulltext indexes which in turn blocks loads of other statements. Trapping this in a reasonable amount of time so that it doesn't affect users too much has been a problem. I then found this 
Tony Rogerson article which went a great way to implementing a solution. 
Repeating Tony's article (look there for the full details, this is just the code)->
First create the trace threshold. Show advance options needs to be enabled:
sp_configure 'show advanced options', 1
go
reconfigure
go
sp_configure
go
-- Set up report capture threshold. Set to 10 seconds here - change figure
-- to suit needs.
sp_configure 'blocked process threshold', 10
go
reconfigure
go
Next create a database to receive the messages:
--CREATE DATABASE RBT.DBAEventManagement
--go
 
USE [RBT.DBAEventManagement]
go
ALTER DATABASE [RBT.DBAEventManagement] SET ENABLE_BROKER
go
CREATE QUEUE syseventqueue 
go
CREATE SERVICE syseventservice
    ON QUEUE syseventqueue ( [http://schemas.microsoft.com/SQL/Notifications/PostEventNotification] )
go
CREATE EVENT NOTIFICATION notify_locks
    ON SERVER
    WITH fan_in
    FOR blocked_process_report
    TO SERVICE 'syseventservice', 'current database';
    
GO
This now enough to capture the messages and queue them up. The next part is to receive the messages and to email out/store history. Only 1 email is sent per block - don't want an email every 10 seconds ! The history table is as follows
USE [RBT.DBAEventManagement]
GO
IF OBJECT_ID('dbo.tbl_messages', 'U') IS NOT NULL
  DROP TABLE dbo.tbl_messages
GO
CREATE TABLE dbo.tbl_messages
(
 Message_Body            XML NOT NULL,
 Message_Sequence_Number INT NOT NULL,
 xactid                  VARCHAR(20) NOT NULL,
 Emailed                 VARCHAR(1) NOT NULL CONSTRAINT [DF_tbl_messages_Emailed]  DEFAULT 'N',
 TMStamp                 SMALLDATETIME NOT NULL CONSTRAINT [DF_tbl_messages_TMStamp]  DEFAULT (GETDATE()),
 CONSTRAINT PK_messages PRIMARY KEY NONCLUSTERED(TMStamp ASC, Message_Sequence_Number ASC)
  )
GO
The stored procedure to receive the messages is as follows:
CREATE  PROCEDURE [dbo].[proc_capture_queue] 
AS
/* ========================================================================
proc_capture_queue
Capture service broker queue and insert into history table.
Email new message
===========================================================================
Version    Date     Author         Comment
-------    ----     ------         -------
 v0.00   14/01/2010 Brian Jones    Initial Version
=========================================================================== */
SET NOCOUNT ON
DECLARE @msg_body       XML,
        @DatabaseID     INT,
        @Process        XML,
        @xactid         VARCHAR(20),
        @emailBody      NVARCHAR(MAX),
        @13LineFeed10   NVARCHAR(2)
        
DECLARE @msgs TABLE (   message_body XML NOT NULL,
                        message_sequence_number INT NOT NULL,
                        xactid       VARCHAR(20))
SET @13LineFeed10 = CHAR(13) + CHAR(10)
IF NOT EXISTS (SELECT * FROM sys.server_event_notifications)
BEGIN
    EXEC msdb.dbo.sp_send_dbmail @profile_name = 'MailProfile',
                                 @recipients = N'myemail@myaddress',
                                 @subject = 'WARNING: SQL Blocking Service Queue',
                                 @body = 'The SQL message queue for tracking blocking sessions is missing, please investigate.'
    RETURN
END;
RECEIVE message_body, message_sequence_number,
        CAST( message_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocked-process[1]/process[1]/@xactid', 'varchar(20)')
FROM syseventqueue
INTO @msgs;
INSERT INTO dbo.tbl_messages(Message_Body, Message_Sequence_Number, xactid) 
SELECT Message_Body, Message_Sequence_Number, xactid
  FROM @msgs
 ORDER BY message_sequence_number
 
DECLARE email_cur CURSOR FOR
  SELECT Message_Body, xactid
    FROM @msgs
   ORDER BY message_sequence_number
OPEN email_cur
FETCH NEXT FROM email_cur INTO @msg_body, @xactid
WHILE @@FETCH_STATUS = 0
BEGIN
  IF NOT EXISTS(SELECT * FROM tbl_messages WHERE xactid = @xactid AND Emailed = 'Y')
  BEGIN
    SELECT @DatabaseId = CAST( @msg_body AS XML ).value( '(/EVENT_INSTANCE/DatabaseID)[1]', 'int' )
    SELECT @emailBody = N'The following process is being blocked.' + @13LineFeed10 + @13LineFeed10 +
                         'Blocked Process '   + @13LineFeed10 +
                         '=============== '   + @13LineFeed10 +
                         'Spid            : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocked-process[1]/process[1]/@spid', 'varchar(10)')
                                              + @13LineFeed10 +
                         'HostPid         : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocked-process[1]/process[1]/@hostpid', 'varchar(10)')
                                              + @13LineFeed10 +
                         'HostName        : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocked-process[1]/process[1]/@hostname', 'varchar(100)')
                                              + @13LineFeed10 +
                         'LoginName       : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocked-process[1]/process[1]/@loginname', 'varchar(100)')
                                              + @13LineFeed10 +
                         'DatabaseName    : ' + DB_Name(@DatabaseID) 
                                              + @13LineFeed10 +
                         'Batch Started   : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocked-process[1]/process[1]/@lastbatchstarted', 'varchar(25)')
                                              + @13LineFeed10 +
                                              + @13LineFeed10 +
                         'Blocking Process '  + @13LineFeed10 +
                         '================ '  + @13LineFeed10 +
                         'Spid            : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocking-process[1]/process[1]/@spid', 'varchar(10)')
                                              + @13LineFeed10 +
                         'HostPid         : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocking-process[1]/process[1]/@hostpid', 'varchar(10)')
                                              + @13LineFeed10 +
                         'HostName        : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocking-process[1]/process[1]/@hostname', 'varchar(100)')
                                              + @13LineFeed10 +
                         'LoginName       : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocking-process[1]/process[1]/@loginname', 'varchar(100)')
                                              + @13LineFeed10 +
                         'DatabaseName    : ' + DB_Name(@DatabaseID) 
                                              + @13LineFeed10 +
                         'Batch Started   : ' + CAST( @msg_body AS XML ).value( '/EVENT_INSTANCE[1]/TextData[1]/blocked-process-report[1]/blocking-process[1]/process[1]/@lastbatchstarted', 'varchar(25)')
                                              + @13LineFeed10 +
                                              + @13LineFeed10 +
                         'Message in full : ' + @13LineFeed10 +
                         CAST(@msg_body AS VARCHAR(MAX))
                         
    EXEC msdb.dbo.sp_send_dbmail @profile_name = 'MailProfile',
                                 @recipients = N'myemail@myaddress',
                                 @subject = 'SQL Blocking Report',
                                 @body = @emailBody
    
    UPDATE tbl_messages
       SET Emailed = 'Y'
     WHERE xactid = @xactid
  END
  
  FETCH NEXT FROM email_cur INTO @msg_body, @xactid
END
CLOSE email_cur
DEALLOCATE email_cur
------------------------ END OF PROCEDURE --------------------------------
OK, I've used a SQL cursor but it's on a small memory table so it isn't an issue.
This stored procedure is in a job which executes every couple of minutes (uses WAITFOR DELAY 00:01 to loop continuously). There is another job which clears the history table down every week.
This is now happily whirring away waiting for the blocking process. This may need (unfortunately) expanding to run a SQL Trace also, so that the blocking process can be investigated further. But that's another blog post !