gpt4 book ai didi

push-notification - PushSharp 的 StopAllServices() 作为 Windows 服务挂起

转载 作者:行者123 更新时间:2023-12-03 01:26:14 24 4
gpt4 key购买 nike

我正在尝试让 PushSharp(2.0.4.0?)作为 Windows 服务运行。每次服务运行并执行push.QueueNotification() 时,小程序都会挂起对stopAPNS(push) 的调用。当我运行类似的代码作为控制台应用程序时,它每次都运行良好(即我正在收到推送通知)。控制台应用程序基于 PushSharp.Sample 项目。

有趣的是,如果我启动/停止我的小程序而不执行任何push.QueueNotification()调用,我的服务将正确退出。在其他情况下,OnStop() 中对 stopAPNS(push) 的调用不会挂起。我想这是有道理的......队列是空的

我在.NET 4.0和4.5下编译了PushSharp项目。在每种情况下,我都会得到相同的行为。

我在下面提供了代码的清理版本。

对于为什么当作为 Windows 服务运行时,push.StopAllServices(true) 调用会挂起有什么想法吗?

谢谢。

public class PushNoteService : System.ServiceProcess.ServiceBase
{
// Initialize global instance of PushBroker service
private PushBroker push = new PushBroker();

#region Constructor
public PushNoteService()
{
// This call is required by the Windows.Forms Component Designer.
InitializeComponent();

// TODO: Add any initialization after the InitComponent call
}
#endregion

#region Component Designer generated code

// The main entry point for the process
static void Main()
{
System.ServiceProcess.ServiceBase[] ServicesToRun;

// More than one user Service may run within the same process. To add
// another service to this process, change the following line to
// create a second service object. For example,
//
// ServicesToRun = new System.ServiceProcess.ServiceBase[] {new PushNoteService(), new MySecondUserService()};
//
ServicesToRun = new System.ServiceProcess.ServiceBase[] { new PushNoteService() };

System.ServiceProcess.ServiceBase.Run(ServicesToRun);
}

#endregion

#region OnStart
/// <summary>
/// Set things in motion so your service can do its work.
/// </summary>
protected override void OnStart(string[] args)
{
YYLog.Log.Instance.Info("Starting service.");

timer2.Enabled = true;

YYLog.Log.Instance.Info("Starting APNS.");
startAPNS(push);
}
#endregion

#region OnStop
/// <summary>
/// Stop this service.
/// </summary>
protected override void OnStop()
{
YYLog.Log.Instance.Info("Stopping service.");

// Add code here to perform any tear-down necessary to stop your service.
timer2.Enabled = false;

YYLog.Log.Instance.Info("Stopping APNS.");
stopAPNS(push);

// some clean up.
push = null;

YYLog.Log.Instance.Info("Service stopped.");
}
#endregion


#region Acess Methods


/// <summary>
/// On Timer_Elasped events, websites are accessed to check status.
/// </summary>
/// <param name="sender">Sender.</param>
/// <param name="e">E.</param>
private void timer2_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
int count = checkForPushRequest();
YYLog.Log.Instance.Info(" Count: " + count.ToString());
if (count > 0)
{
stopAPNS(push);
}

}

private int checkForPushRequest()
{
YYLog.Log.Instance.Info("Processing push notifications...");

int count = 0;

// Get the ConnectionStrings collection.
ConnectionStringSettings connections = ConfigurationManager.ConnectionStrings["MyDB"];

using (SqlConnection conn = new SqlConnection(connections.ConnectionString))
{
conn.Open();

SqlCommand cmd = new SqlCommand("MySP", conn);
cmd.CommandType = System.Data.CommandType.StoredProcedure;

SqlDataReader dr = cmd.ExecuteReader(System.Data.CommandBehavior.CloseConnection);

while (dr.Read())
{
// increment counter
count++;

int badgeNumber = 0;

string deviceToken = Convert.ToString(dr[dr.GetOrdinal("DeviceToken")]);
string alertMessage = Convert.ToString(dr[dr.GetOrdinal("AlertMessage")]);
if (!dr.IsDBNull(dr.GetOrdinal("BadgeNumber")))
{
badgeNumber = Convert.ToInt16(dr[dr.GetOrdinal("BadgeNumber")]);
}
string soundFile = Convert.ToString(dr[dr.GetOrdinal("SoundFile")]);

// Send out the notification to APNS
YYLog.Log.Instance.Trace(" Sending notification to " + deviceToken);
sendPush(deviceToken, alertMessage, badgeNumber, soundFile);
}

dr.Close();
}

return count;
}

private void sendPush(string DeviceToken, string AlertMessage, int BadgeNumber, string SoundFile)
{
push.QueueNotification(new AppleNotification()
.ForDeviceToken(DeviceToken)
.WithAlert(AlertMessage)
.WithBadge(BadgeNumber)
.WithSound(SoundFile));

}

private void startAPNS(PushBroker push)
{
//Wire up the events for all the services that the broker registers
push.OnNotificationSent += NotificationSent;
push.OnChannelException += ChannelException;
push.OnServiceException += ServiceException;
push.OnNotificationFailed += NotificationFailed;
push.OnDeviceSubscriptionExpired += DeviceSubscriptionExpired;
push.OnDeviceSubscriptionChanged += DeviceSubscriptionChanged;
push.OnChannelCreated += ChannelCreated;
push.OnChannelDestroyed += ChannelDestroyed;

string appleCertFileName = System.Configuration.ConfigurationManager.AppSettings["APNS_Certificate"];
var appleCert = File.ReadAllBytes(appleCertFileName);

string appleCertPassword = System.Configuration.ConfigurationManager.AppSettings["APNS_Certificate_Password"];
bool productionMode = bool.Parse(System.Configuration.ConfigurationManager.AppSettings["APNS_Production_Mode"]);
push.RegisterAppleService(new ApplePushChannelSettings(productionMode, appleCert, appleCertPassword)); //Extension method
}

private void stopAPNS(PushBroker push)
{
YYLog.Log.Instance.Info("Waiting for Queue to Finish...");

//Stop and wait for the queues to drains
push.StopAllServices(true);

YYLog.Log.Instance.Info("Queue Finished");
}

#region Events
private void DeviceSubscriptionChanged(object sender, string oldSubscriptionId, string newSubscriptionId, INotification notification)
{
//Currently this event will only ever happen for Android GCM
YYLog.Log.Instance.Info("Device Registration Changed: Old-> " + oldSubscriptionId + " New-> " + newSubscriptionId + " -> " + notification);
}

private void NotificationSent(object sender, INotification notification)
{
YYLog.Log.Instance.Info("Sent: " + sender + " -> " + notification);
}

private void NotificationFailed(object sender, INotification notification, Exception notificationFailureException)
{
YYLog.Log.Instance.Error("Failure: " + sender + " -> " + notificationFailureException.Message + " -> " + notification);
}

private void ChannelException(object sender, IPushChannel channel, Exception exception)
{
YYLog.Log.Instance.Error("Channel Exception: " + sender + " -> " + exception);
}

private void ServiceException(object sender, Exception exception)
{
YYLog.Log.Instance.Error("Channel Exception: " + sender + " -> " + exception);
}

private void DeviceSubscriptionExpired(object sender, string expiredDeviceSubscriptionId, DateTime timestamp, INotification notification)
{
YYLog.Log.Instance.Info("Device Subscription Expired: " + sender + " -> " + expiredDeviceSubscriptionId);
}

private void ChannelDestroyed(object sender)
{
YYLog.Log.Instance.Info("Channel Destroyed for: " + sender);
}

private void ChannelCreated(object sender, IPushChannel pushChannel)
{
YYLog.Log.Instance.Info("Channel Created for: " + sender);
}
#endregion

#endregion

}

最佳答案

我们有一个 Web 应用程序,但我认为解决方案可以是相同的。我们花了一整天的时间来猜测这个问题!最后是Newtonsoft.Json版本错误我们的解决方案中的一些项目依赖于该库的旧版本,因此我们运气不好,在 Web 项目的/bin 文件夹中获得了错误的版本。

关于push-notification - PushSharp 的 StopAllServices() 作为 Windows 服务挂起,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16713546/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com