无法使用 SqlDependency 接收更改通知



我是第一次使用Sql Server Service Broker,尝试在特定表中发生更改时订阅通知。当我打电话给command.ExecuteReader()时,我收到以下异常:

System.InvalidOperationException :当使用 SqlDependency 而不提供选项值时,必须在执行添加到 SqlDependency 实例的命令之前调用 SqlDependency.Start((。

我创建了一个测试来重现场景(为简洁起见,省略了一些相关方法(,如下所示:

private string _queueName = "EventsToPublishChangeMessages";
private bool _notificationReceived;
[Test]
public void WhyDoesExceptionIndcateSqlDependencyStartHasNotBeenCalledPriorToCommandExecuteReader()
{
Console.WriteLine($"canRequestNotifications: {CanRequestNotifications()}"); // returns true
var connectionString = GetConnectionString();
var started = SqlDependency.Start(connectionString, _queueName); // exception below seems to suggest that I haven't started the SqlDependency. Am I doing something wrong on this line?
Console.WriteLine($"Started:{started}"); // returns true
var connection = new SqlConnection(connectionString);
var command = new SqlCommand("SELECT Id, EventType, [Data], Created FROM dbo.EventsToPublish", connection);
var sqlDependency = new SqlDependency(command);
sqlDependency.OnChange += OnChange;
connection.Open();
// The following line causes the exception:
// System.InvalidOperationException : When using SqlDependency without providing an options value, SqlDependency.Start() must be called prior to execution of a command added to the SqlDependency instance.
// But you can see that I *did* call SqlDependency.Start() above.
using (var reader = command.ExecuteReader())
{
while (reader.Read())
{
Process(reader);
}
}
TryToTriggerANotification(connectionString);
Thread.Sleep(5000); // ie. wait a few seconds to ensure notification-handling background thread has had a chance to complete.
Assert.That(_notificationReceived, Is.True);
}
private void TryToTriggerANotification(string connectionString)
{
using (var connection = new SqlConnection(connectionString))
{
using (var command = new SqlCommand(
"INSERT INTO dbo.EventsToPublish(Id, EventType, [Data], Created) VALUES(newid(), 'StackOverflowQuestionTest', '', GETDATE())",
connection))
{
connection.Open();
command.ExecuteNonQuery();
}
}
}
private void OnChange(object sender, SqlNotificationEventArgs e)
{
_notificationReceived = true;
}
private bool CanRequestNotifications()
{
try
{
var sqlClientPermission = new SqlClientPermission(PermissionState.Unrestricted);
sqlClientPermission.Demand();
return true;
}
catch
{
return false;
}
}

我发出了以下 SQL 查询,以尝试更好地了解正在发生的事情:

select * from sys.service_queues -- can see my EventsToPublishChangeMessages queue
select is_broker_enabled from sys.databases where database_id=db_id() -- returns 1
select * from sys.dm_qn_subscriptions -- returns nothing
select * from sys.transmission_queue -- returns nothing

有没有人知道我可能做错了什么?

错误消息说:

在执行添加到 SqlDependency 实例的命令之前,必须调用SqlDependency.Start()

SqlDependency.Start的帮助指出:

启动用于接收依赖项更改通知的侦听器。

确保每个应用域仅调用一次"开始">

因此,在您的应用程序启动中的某个位置拨打Start()电话。由于这看起来像一种测试方法,也许是一种[ClassInitialize]装饰的方法。

最新更新