在 Android KitKat 中接收彩信

Posted

技术标签:

【中文标题】在 Android KitKat 中接收彩信【英文标题】:Receive MMS messages in Android KitKat 【发布时间】:2014-03-11 23:24:34 【问题描述】:

因此,来自#DevBytes 的这段视频android 4.4 SMS APIs 解释了最近对 KitKat 中 SMS API 的更改。他们还提供了一个示例项目的链接。 http://goo.gl/uQ3Nih

他们建议您在服务中处理彩信的接收。这一切看起来都很好,除了他们忽略了提到最无证的部分。如何实际处理收到的彩信。

这是项目中的示例 https://gist.github.com/lawloretienne/8970938

我已尝试“处理彩信”

https://gist.github.com/lawloretienne/8971050

我可以从 Intent 中获取额外信息,但我可以提取的唯一有意义的东西是发送 MMS 的号码。

任何人都可以为我指出正确的方向吗?

我注意到 WAP_PUSH_MESSAGE 包含一些内容,FROM、SUBJECT 和 CONTENT_LOCATION。

内容位置似乎是包含彩信内容的 url。我怎样才能访问它?

这是该 URL 的示例

https://atl1mmsget.msg.eng.t-mobile.com/mms/wapenc?location=XXXXXXXXXXX_14zbwk&rid=027

X 是我正在测试的设备的电话号码中的一个数字。

T-Mobile 在美国的 MMSC(多媒体消息服务中心)好像是http://mms.msg.eng.t-mobile.com/mms/wapenc

根据这个列表:http://www.activexperts.com/xmstoolkit/mmsclist/

【问题讨论】:

如果你的声望是 158,你怎么能提供 300 的赏金呢? uoy 至少需要另外 142 个声望点。我错过了什么吗? 我之前有458声望,愿意提供大额赏金,因为我很长时间没有遇到任何解决这个问题的方法。 我明白了...积分是在您分配赏金之前获得的。抱歉,我没有赏金经验,但对我来说似乎很奇怪。 ;) @toobsco42 请更新您的死链接。另外,你有没有让这个工作?我发布了一个相关问题here。 【参考方案1】:

文档数量为零,因此这里有一些信息可以提供帮助。

1) com.google.android.mms.pdu 来自源代码。您需要 Pdu 实用程序。

2) 您从传入 mms 广播的字节数组额外获得通知推送 (intent.getByteArrayExtra("data"))。

3) 将通知推送解析成一个GenericPdu (new PduParser(rawPdu).parse())。

4) 您需要 TransactionSettings 来与运营商的 wap 服务器通信。我在下面的#5 之后得到交易设置。我用:

TransactionSettings transactionSettings = new TransactionSettings(mContext, mConnMgr.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_MMS).getExtraInfo());

5) 强制通过 wifi 进行网络通信。我使用以下。

private boolean beginMmsConnectivity() 
    try 
        int result = mConnMgr.startUsingNetworkFeature(ConnectivityManager.TYPE_MOBILE, Phone.FEATURE_ENABLE_MMS);
        NetworkInfo info = mConnMgr.getNetworkInfo(ConnectivityManager.TYPE_MOBILE_MMS);
        boolean isAvailable = info != null && info.isConnected() && result == Phone.APN_ALREADY_ACTIVE && !Phone.REASON_VOICE_CALL_ENDED.equals(info.getReason());
        return isAvailable;
     catch(Exception e) 
        return false;
    

6) 然后您需要确保到主机的路由。

private static void ensureRouteToHost(ConnectivityManager cm, String url, TransactionSettings settings) throws IOException 
    int inetAddr;
    if (settings.isProxySet()) 
        String proxyAddr = settings.getProxyAddress();
        inetAddr = lookupHost(proxyAddr);
        if (inetAddr == -1) 
            throw new IOException("Cannot establish route for " + url + ": Unknown host");
         else 
            if (!cm.requestRouteToHost(ConnectivityManager.TYPE_MOBILE_MMS, inetAddr))
                throw new IOException("Cannot establish route to proxy " + inetAddr);
        
     else 
        Uri uri = Uri.parse(url);
        inetAddr = lookupHost(uri.getHost());
        if (inetAddr == -1) 
            throw new IOException("Cannot establish route for " + url + ": Unknown host");
         else 
            if (!cm.requestRouteToHost(ConnectivityManager.TYPE_MOBILE_MMS, inetAddr))
                throw new IOException("Cannot establish route to " + inetAddr + " for " + url);
        
    

这里是lookupHost方法:

private static int lookupHost(String hostname) 
    InetAddress inetAddress;
    try 
        inetAddress = InetAddress.getByName(hostname);
     catch (UnknownHostException e) 
        return -1;
    
    byte[] addrBytes;
    int addr;
    addrBytes = inetAddress.getAddress();
    addr = ((addrBytes[3] & 0xff) << 24) | ((addrBytes[2] & 0xff) << 16) | ((addrBytes[1] & 0xff) << 8) | (addrBytes[0] & 0xff);
    return addr;

我还喜欢使用基于反射的方法来改进 ensureRouteToHost 功能:

private static void ensureRouteToHostFancy(ConnectivityManager cm, String url, TransactionSettings settings) throws IOException, NoSuchMethodException, IllegalAccessException, IllegalArgumentException, InvocationTargetException 
    Method m = cm.getClass().getMethod("requestRouteToHostAddress", new Class[]  int.class, InetAddress.class );
    InetAddress inetAddr;
    if (settings.isProxySet()) 
        String proxyAddr = settings.getProxyAddress();
        try 
            inetAddr = InetAddress.getByName(proxyAddr);
         catch (UnknownHostException e) 
            throw new IOException("Cannot establish route for " + url + ": Unknown proxy " + proxyAddr);
        
        if (!(Boolean) m.invoke(cm, new Object[]  ConnectivityManager.TYPE_MOBILE_MMS, inetAddr ))
            throw new IOException("Cannot establish route to proxy " + inetAddr);
     else 
        Uri uri = Uri.parse(url);
        try 
            inetAddr = InetAddress.getByName(uri.getHost());
         catch (UnknownHostException e) 
            throw new IOException("Cannot establish route for " + url + ": Unknown host");
        
        if (!(Boolean) m.invoke(cm, new Object[]  ConnectivityManager.TYPE_MOBILE_MMS, inetAddr ))
            throw new IOException("Cannot establish route to " + inetAddr + " for " + url);
    

7) 在确保到主机的路由之后,您可以从源代码中获取 HttpUtls。我使用 OkHttp 大量修改了我的实现以改进通信。

byte[] rawPdu = HttpUtils.httpConnection(mContext, mContentLocation, null, HttpUtils.HTTP_GET_METHOD, mTransactionSettings.isProxySet(), mTransactionSettings.getProxyAddress(), mTransactionSettings.getProxyPort());

8) 从生成的字节数组中使用 PduParser 解析 GenericPdu。然后您可以提取正文并转换为 MultimediaMessagePdu。

9) 然后你可以迭代 PDU 的各个部分。

使用彩信需要考虑的事项数不胜数。想到的一件事是幻灯片有多烦人,所以我要做的是检测 PDU 中是否有超过 1 个部分,然后我复制标头并创建单独的 MultimediaMessagePdu,我将它们分别保存到手机的 mms 内容提供商.不要忘记复制标题,特别是如果您支持群组消息传递。群消息是另一回事,因为 PDU 中的传入电话号码并不能说明全部情况(MultimediaMessagePdu.mmpdu())。您使用以下代码提取的标题中有更多联系人。

private HashSet<String> getRecipients(GenericPdu pdu) 
    PduHeaders header = pdu.getPduHeaders();
    HashMap<Integer, EncodedStringValue[]> addressMap = new HashMap<Integer, EncodedStringValue[]>(ADDRESS_FIELDS.length);
    for (int addrType : ADDRESS_FIELDS) 
        EncodedStringValue[] array = null;
        if (addrType == PduHeaders.FROM) 
            EncodedStringValue v = header.getEncodedStringValue(addrType);
            if (v != null) 
                array = new EncodedStringValue[1];
                array[0] = v;
            
         else 
            array = header.getEncodedStringValues(addrType);
        
        addressMap.put(addrType, array);
    
    HashSet<String> recipients = new HashSet<String>();
    loadRecipients(PduHeaders.FROM, recipients, addressMap, false);
    loadRecipients(PduHeaders.TO, recipients, addressMap, true);
    return recipients;

这是加载接收者的方法:

private void loadRecipients(int addressType, HashSet<String> recipients, HashMap<Integer, EncodedStringValue[]> addressMap, boolean excludeMyNumber) 
    EncodedStringValue[] array = addressMap.get(addressType);
    if (array == null) 
        return;
    
    // If the TO recipients is only a single address, then we can skip loadRecipients when
    // we're excluding our own number because we know that address is our own.
    if (excludeMyNumber && array.length == 1) 
        return;
    
    String myNumber = excludeMyNumber ? mTelephonyManager.getLine1Number() : null;
    for (EncodedStringValue v : array) 
        if (v != null) 
            String number = v.getString();
            if ((myNumber == null || !PhoneNumberUtils.compare(number, myNumber)) && !recipients.contains(number)) 
                // Only add numbers which aren't my own number.
                recipients.add(number);
            
        
    

这是迭代 MultimediaMessagePdu 部分的方法。

private void processPduAttachments() throws Exception 
    if (mGenericPdu instanceof MultimediaMessagePdu) 
        PduBody body = ((MultimediaMessagePdu) mGenericPdu).getBody();
        if (body != null) 
            int partsNum = body.getPartsNum();
            for (int i = 0; i < partsNum; i++) 
                try 
                    PduPart part = body.getPart(i);
                    if (part == null || part.getData() == null || part.getContentType() == null || part.getName() == null)
                        continue;
                    String partType = new String(part.getContentType());
                    String partName = new String(part.getName());
                    Log.d("Part Name: " + partName);
                    Log.d("Part Type: " + partType);
                    if (ContentType.isTextType(partType)) 
                     else if (ContentType.isImageType(partType)) 
                     else if (ContentType.isVideoType(partType)) 
                     else if (ContentType.isAudioType(partType)) 
                    
                 catch (Exception e) 
                    e.printStackTrace();
                    // Bad part shouldn't ruin the party for the other parts
                
            
        
     else 
        Log.d("Not a MultimediaMessagePdu PDU");
    

还有更多考虑因素,例如动画 GIF 支持,这是完全可能的 :) 一些运营商也支持确认报告和交付报告,除非用户真的想要 mms 交付报告,否则您很可能会忽略这些 wap 通信。

【讨论】:

嘿@Noah,android 源代码中有大量与mms 相关的类。我需要一组特定的课程才能使其正常工作吗?因为有大量依赖项需要通过无效导入来解决。 您主要需要 pdu 实用程序。 com.google.android.mms.pdu。您将需要 TransactionSettings 及其依赖项,它有一些问题,需要在许多方面进行大量简化,至少是内容提供程序查询的投影和 where 子句。请记住,事务设置中的查询只能在 KitKat 设备上可靠地工作。 我不理解 ADDRESS_FIELDS.length 等常量的使用,你能澄清一下这些应该被初始化成什么吗? @TimMiller 它们是从源文件之一导入的值。 private static final int[] ADDRESS_FIELDS = new int[] PduHeaders.BCC, PduHeaders.CC, PduHeaders.FROM, PduHeaders.TO ;

以上是关于在 Android KitKat 中接收彩信的主要内容,如果未能解决你的问题,请参考以下文章

我想要来自 android 中 MMS db 的彩信号码、发送/接收、正文和附件详细信息...怎么做?

解决:关闭移动数据,接收彩信,不能成功接收下载彩信

为啥 Json 对象顺序在 android KITKAT 及以下版本中混淆?

如何在 Android Kitkat 中设置我的短信应用默认值?

android捕获照片并将其保存在图像视图中并发送彩信?

如何将图像从 Android 中的 Drawable 附加到彩信?