APNS 接受推送消息,但它们永远不会成为目标

Posted

技术标签:

【中文标题】APNS 接受推送消息,但它们永远不会成为目标【英文标题】:APNS accepts push messages, but they are never come to target 【发布时间】:2015-01-29 16:57:05 【问题描述】:

我正在尝试为我的后端服务器服务添加苹果推送通知,但我遇到了一个我无法理解的问题。 我正在使用 Pushy 库 (com.relayriders.pushy)。 我的代码就像他们的 github 页面推荐的一样。 代码是有效的,没有例外,push 看起来像是正确形成的。推送发送到 APNS,但永远不会到达设备。 设备令牌和证书是正确的,我的朋友通过他的测试程序向目标设备发送了推送。 我还测试了 com.notnoop.apns 库,结果相同 - 没有例外,仍然没有推送设备 这是我的发件人类:

package ua.asprelis.communicator.push;

import com.relayrides.pushy.apns.ApnsEnvironment;
import com.relayrides.pushy.apns.FailedConnectionListener;
import com.relayrides.pushy.apns.PushManager;
import com.relayrides.pushy.apns.PushManagerConfiguration;
import com.relayrides.pushy.apns.RejectedNotificationListener;
import com.relayrides.pushy.apns.RejectedNotificationReason;
import com.relayrides.pushy.apns.util.ApnsPayloadBuilder;
import com.relayrides.pushy.apns.util.MalformedTokenStringException;
import com.relayrides.pushy.apns.util.SSLContextUtil;
import com.relayrides.pushy.apns.util.SimpleApnsPushNotification;
import com.relayrides.pushy.apns.util.TokenUtil;
import java.io.IOException;
import java.net.URL;
import java.net.URLDecoder;
import java.security.KeyManagementException;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.security.UnrecoverableKeyException;
import java.security.cert.CertificateException;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.net.ssl.SSLHandshakeException;


public class ApplePushNotifier

    private final PushManager<SimpleApnsPushNotification> pushManager;

    private final ApnsPayloadBuilder payloadBuilder = new ApnsPayloadBuilder();

    public ApplePushNotifier() throws KeyStoreException, NoSuchAlgorithmException, CertificateException, UnrecoverableKeyException, KeyManagementException, IOException 
        URL url = getClass().getResource("/certpath/cert.p12");
        this.pushManager = new PushManager<>(
            ApnsEnvironment.getSandboxEnvironment(),
            SSLContextUtil.createDefaultSSLContext(URLDecoder.decode(url.getPath(), "UTF-8"), "password"),
            null, // Optional: custom event loop group
            null, // Optional: custom ExecutorService for calling listeners
            null, // Optional: custom BlockingQueue implementation
            new PushManagerConfiguration(),
            "ExamplePushManager");
        pushManager.start();
        pushManager.registerRejectedNotificationListener(new MyRejectedNotificationListener());
        pushManager.registerFailedConnectionListener(new MyFailedConnectionListener());
    

    public void sendpush(String message, byte[] token) throws InterruptedException 
        String payload = payloadBuilder.setAlertBody(message).setSoundFileName("ring-ring.aiff").buildWithDefaultMaximumLength();
        pushManager.getQueue().put(new SimpleApnsPushNotification(token, payload));
    

    public void sendpush(String message, String stoken) throws InterruptedException, MalformedTokenStringException 
        byte[]token = TokenUtil.tokenStringToByteArray(stoken);
        String payload = payloadBuilder.setAlertBody(message).setSoundFileName("ring-ring.aiff").buildWithDefaultMaximumLength();
        SimpleApnsPushNotification notification = new SimpleApnsPushNotification(token, payload);
        pushManager.getQueue().put(notification);
        System.out.println("Queued: "+notification);
    

    public void closeSender() 
        if(pushManager!=null) 
            try 
                pushManager.shutdown();
             catch (InterruptedException ex) 
                Logger.getLogger(ApplePushNotifier.class.getName()).log(Level.SEVERE, null, ex);
            
        
    

    private class MyRejectedNotificationListener implements RejectedNotificationListener<SimpleApnsPushNotification> 
        @Override
        public void handleRejectedNotification(
            final PushManager<? extends SimpleApnsPushNotification> pushManager,
            final SimpleApnsPushNotification notification,
            final RejectedNotificationReason reason) 
            System.out.format("%s was rejected with rejection reason %s\n", notification, reason);
        
    

    private class MyFailedConnectionListener implements FailedConnectionListener<SimpleApnsPushNotification> 
        @Override
        public void handleFailedConnection(
            final PushManager<? extends SimpleApnsPushNotification> pushManager,
            final Throwable cause) 
            if (cause instanceof SSLHandshakeException)
                // This is probably a permanent failure, and we should shut down the PushManager.
            else
                System.out.println(cause);
        
    

以及运行推送的测试类:

@Test
public void testPushMessage() 
    Random random = new Random();
    String testMessage = "Hello Test! "+random.nextInt(1000);
    String testToken = "e6878d3993abfaec48220b9d4d3ea0b576c22351c7fbbb5faeb5449bf7f24452";
                      //e6878d39 93abfaec 48220b9d 4d3ea0b5 76c22351 c7fbbb5f aeb5449b f7f24452

    ApplePushNotifier notifier=null;
    try 
        notifier = new ApplePushNotifier();
        notifier.sendpush(testMessage, testToken);
     catch(Exception ex) 
        Logger.getLogger(applePushNotifierTest.class.getName()).log(Level.SEVERE, null, ex);
     finally 
        if (notifier!=null)
            notifier.closeSender();
    

我的输出:

[main] INFO com.relayrides.pushy.apns.PushManager - ExamplePushManager starting.
    Queued: SimpleApnsPushNotification [token=[-26, -121, -115, 57, -109, -85, -6, -20, 72, 34, 11, -99, 77, 62, -96, -75, 118, -62, 35, 81, -57, -5, -69, 95, -82, -75, 68, -101, -9, -14, 68, 82], payload="aps":"alert":"Hello Test! 955","sound":"ring-ring.aiff", invalidationTime=null, priority=IMMEDIATE]
    [main] INFO com.relayrides.pushy.apns.PushManager - ExamplePushManager shutting down.

我觉得问题出在一些小块上,我看不到。如果两个测试库都不起作用,那就奇怪了

【问题讨论】:

您是使用开发配置配置文件还是生产配置配置文件来测试您的应用程序? 是的,我使用了生产证书,但是将我的发件人类设置为沙盒模式。谢谢 你很幸运,我上周犯了同样愚蠢的错误,所以我仍然记忆犹新:) 谁能告诉我SimpleApnsPushNotification是什么。我收到一个错误:SimpleApnsPushNotification 无法解析为类型 我正在使用 maven 依赖 &lt;dependency&gt; &lt;groupId&gt;com.relayrides&lt;/groupId&gt; &lt;artifactId&gt;pushy&lt;/artifactId&gt; &lt;version&gt;0.9.2&lt;/version&gt; &lt;/dependency&gt; 但没有类似的类:com.relayrides.pushy.apns.PushManager 【参考方案1】:

正如我所说 - 我犯了一个愚蠢的错误。我真是个笨蛋。我用过

ApnsEnvironment.getSandboxEnvironment()

而不是

ApnsEnvironment.getProductionEnvironment()

在两个库中。

那是我第一次使用 APNS。

【讨论】:

您好,我收到此错误构造函数 PushManager(ApnsEnvironment, SSLContext, NioEventLoopGroup, ExecutorService, BlockingQueue, PushManagerConfiguration, String) 指的是缺少的类型 NioEventLoopGroup 关于这个错误有什么想法吗? 好的,我得到了正确的 netty jar 文件没有添加到我的类路径中 你使用了什么maven依赖版本或jar?

以上是关于APNS 接受推送消息,但它们永远不会成为目标的主要内容,如果未能解决你的问题,请参考以下文章

APNS 推送请求已成功发送,但通行证未更新

react-native 使用leanclound消息推送

极光推送流程

关于项目中的推送问题。。

用HTML5开发的WebApp怎么实现消息推送

无法接收生产 apns 的推送消息