IOS 推送訊息 php做推送服務端

子非あ魚發表於2013-08-28

IOS推送訊息是許多IOS應用都具備的功能,最近也在研究這個功能,參考了很多資料終於搞定了,下面就把步驟拿出來分享下:
 
 
 
iOS訊息推送的工作機制可以簡單的用下圖來概括:

 
Provider是指某個iPhone軟體的Push伺服器,APNS是Apple Push Notification Service的縮寫,是蘋果的伺服器。
 
上圖可以分為三個階段:
第一階段:應用程式把要傳送的訊息、目的iPhone的標識打包,發給APNS。 
第二階段:APNS在自身的已註冊Push服務的iPhone列表中,查詢有相應標識的iPhone,並把訊息傳送到iPhone。 
第三階段:iPhone把發來的訊息傳遞給相應的應用程式,並且按照設定彈出Push通知。

 
從上圖我們可以看到:
1、應用程式註冊訊息推送。
2、iOS從APNS Server獲取device token,應用程式接收device token。
3、應用程式將device token傳送給PUSH服務端程式。
4、服務端程式向APNS服務傳送訊息。
5、APNS服務將訊息傳送給iPhone應用程式。
 
無論是iPhone客戶端和APNS,還是Provider和APNS,都需要通過證照進行連線。
 
下面我介紹一下幾種用到的證照。
 
一、CSR檔案
 
1、生成Certificate Signing Request(CSR)

 
2、填寫你的郵箱和常用名稱,並選擇儲存到硬碟。

 
點選繼續:

 
這樣就在本地生成了一個Push.certSigningRequest檔案。
 
二、p12檔案
 
1、匯出金鑰。


 
2、輸入你的密碼。
 

 
這樣就生成了一個Push.p12檔案。
 
三、SSL certificate檔案
 
1、用你付過費的帳號登入到iOS Provisioning Portal,並新建一個App ID,這個過程可以參考:iOS應用的真機除錯,這樣就會生成下面這條記錄:

 
2、點選右側的Configure:

 
3、點選Development Push SSL Certificate一行後的Configure:
 

 
4、點選Continue:

 
5、選擇前面生成好的Push.certSigningRequest檔案,點選Generate,出現如下所示的頁面:

 
6、點選Continue:

 
 
7、點選Download,並將檔案命名為aps_developer_identity.cer。
 
8、點選Done,你會發現狀態變成了Enabled:

 
 
到現在為止,我們已經生成了三個檔案:
1、Push.certSigningRequest
2、Push.p12
3、aps_developer_identity.cer
 
雙擊aps_developer_dientity.cer 註冊到你的鑰匙串中,這樣你的鑰匙串中就會有

二、準備profile證照,因為推送訊息只能再真機上測試,所以要建一個profile證照
點選"new profile"為上面新建的APP ID建個profile ,成功之後下載*_Dev_Profile.mobileprovision
雙擊將其加入到xcode 的Provisioning Profiles 中,這裡有一點要注意,再將這個加入xcode之前如果之前已經加入過一定要把之前加入的刪掉,如果有多個的話會出錯。
 
三、工程程式碼
到這裡證照已經準備完畢,接下來,我們在xcode中新建一個測試工程,注意設定工程的Bundle Identifier必須與上面建的APP ID 裡的相同

在didFinishLaunchingWithOptions 中加入一下程式碼
 
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
 
[self.window makeKeyAndVisible];
    [[UIApplication sharedApplication] registerForRemoteNotificationTypes: UIRemoteNotificationTypeBadge |UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert];
    return YES;
 
}
 
 
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)pToken {

    NSLog(@"regisger success:%@", pToken);
    
    //註冊成功,將deviceToken儲存到應用伺服器資料庫中
    
}

 
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo{
    // 處理推送訊息
    UIAlertView *alert=[[UIAlertView alloc]initWithTitle:@"通知" message:@"我的資訊" delegate:selfcancelButtonTitle:@"取消" otherButtonTitles:nil, nil];
    [alert show];
    [alert release];
NSLog(@"%@", userInfo);
}

- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error {
    NSLog(@"Regist fail%@",error); 
    

  
}
 
到這裡一切順利的話我們就可以在真機執行了,註冊成功我們會得到iphone 的deviceToken,
 
My token is:
<740f4707 bebcf74f 9b7c25d4 8e335894 5f6aa01d a5ddb387 462c7eaf 61bb78ad>
 
四、在應用伺服器採用php的方式將訊息推送給APNS,
1、php連線APNS也是需要證照的,還記得我們上面獲得的幾個證照嗎?開啟終端,對上面的證照做如下處理,
cd  進入證照所在目錄
 
把.cer檔案轉換成.pem檔案:
$ openssl x509 -in aps_developer_identity.cer -inform der
-out PushChatCert.pem
把私鑰Push.p12檔案轉換成.pem檔案:
$ openssl pkcs12 -nocerts -out PushChatKey.pem -in Push.p12
Enter Import Password:
MAC verified OK
Enter PEM pass phrase:
Verifying – Enter PEM pass phrase:
你首先需要為.p12檔案輸入passphrase密碼短語,這樣OpenSSL可以讀它。然後你需要鍵入一個新的密碼短語來加密PEM檔案。還是使用”pushchat”來作為PEM的密碼短語。你需要選擇一些更安全的密碼短語。
注意:如果你沒有鍵入一個PEM passphrase,OpenSSL將不會返回一個錯誤資訊,但是產生的.pem檔案裡面將不會含有私鑰。
最後。把私鑰和證照整合到一個.pem檔案裡:
$ cat PushChatCert.pem PushChatKey.pem > ck.pem
為了測試證照是否工作,執行下面的命令
$ telnet gateway.sandbox.push.apple.com 2195
Trying 17.172.232.226…
Connected to gateway.sandbox.push-apple.com.akadns.net.
Escape character is ‘^]’.
它將嘗試傳送一個規則的,不加密的連線到APNS服務。如果你看到上面的反饋,那說明你的MAC能夠到達APNS。按下Ctrl+C 關閉連線。如果得到一個錯誤資訊,那麼你需要確保你的防火牆允許2195埠。
然後再次連線,這次用我們的SSL證照和私鑰來設定一個安全的連線:
$ openssl s_client -connect gateway.sandbox.push.apple.com:2195
-cert PushChatCert.pem -key PushChatKey.pem
Enter pass phrase for PushChatKey.pem:
你會看到一個完整的輸出,讓你明白OpenSSL在後臺做什麼。如果連線是成功的,你可以鍵入一些字元。當你按下回車後,服務就會斷開連線。如果在建立連線時有問題,OpenSSL將會給你一個錯誤訊息,
ck.pem檔案就是我們需要得到php連線APNS 的檔案,將ck.pem和push.php放入同一目錄上傳到伺服器,push.php的程式碼如下:
 
<?php

// 這裡是我們上面得到的deviceToken,直接複製過來(記得去掉空格
$deviceToken = '740f4707bebcf74f 9b7c25d4 8e3358945f6aa01da5ddb387462c7eaf 61bb78ad';

// Put your private key's passphrase here:
$passphrase = 'abc123456';

// Put your alert message here:
$message = 'My first push test!';

////////////////////////////////////////////////////////////////////////////////

$ctx = stream_context_create();
stream_context_set_option($ctx, 'ssl', 'local_cert', 'ck.pem');
stream_context_set_option($ctx, 'ssl', 'passphrase', $passphrase);

// Open a connection to the APNS server
//這個為正是的釋出地址
 //$fp = stream_socket_client(“ssl://gateway.push.apple.com:2195“, $err, $errstr, 60, //STREAM_CLIENT_CONNECT, $ctx);
//這個是沙盒測試地址,釋出到appstore後記得修改哦
$fp = stream_socket_client(
'ssl://gateway.sandbox.push.apple.com:2195', $err,
$errstr, 60, STREAM_CLIENT_CONNECT|STREAM_CLIENT_PERSISTENT, $ctx);

if (!$fp)
exit("Failed to connect: $err $errstr" . PHP_EOL);

echo 'Connected to APNS' . PHP_EOL;

// Create the payload body
$body['aps'] = array(
'alert' => $message,
'sound' => 'default'
);

// Encode the payload as JSON
$payload = json_encode($body);

// Build the binary notification
$msg = chr(0) . pack('n', 32) . pack('H*', $deviceToken) . pack('n', strlen($payload)) . $payload;

// Send it to the server
$result = fwrite($fp, $msg, strlen($msg));

if (!$result)
echo 'Message not delivered' . PHP_EOL;
else
echo 'Message successfully delivered' . PHP_EOL;

// Close the connection to the server
fclose($fp);
?>

接下來我們訪問http://localhost/push/push.php

iphone就會接收到一條推送訊息了,如果有問題的話就檢查上面的操作步驟,特別是加紅的部分
 
另外去除標記的方法為,在viewDidApper中加入

int badge = [UIApplication sharedApplication].applicationIconBadgeNumber;
    if(badge > 0)
    {
        badge--;
        [UIApplication sharedApplication].applicationIconBadgeNumber = badge;
    }

相關文章