目标:(注意:所选答案为CDMA (3gpp2) please refer here生成GSM(3gpp)PDU
创建可以传递到SmsMessage.createFromPdu(byte[] pdu)
的PDU。
我正在向我的BroadcastReciever
中的一个收听短信的“广播意图”。
一个BroadcastReciever
将android.provider.Telephony.SMS_RECEIVED
用于“真实”短信
对这些新的“应用程序短信”使用自定义intent-filter
操作。
@Override
public void onReceive(Context context, Intent intent) {
Bundle bundle = intent.getExtras();
if (bundle != null) {
Object[] pdusObj = (Object[]) bundle.get("pdus");
SmsMessage[] messages = new SmsMessage[pdusObj.length];
// getting SMS information from Pdu.
for (int i = 0; i < pdusObj.length; i++) {
messages[i] = SmsMessage.createFromPdu((byte[]) pdusObj[i]);
}
for (SmsMessage currentMessage : messages) {
//the currentMessage.getDisplayOriginatingAddress()
//or .getDisplayMessageBody() is null if I Broadcast a fake sms
Log.i("BB", "address:"+currentMessage.getDisplayOriginatingAddress()+" message:"+currentMessage.getDisplayMessageBody());
...
所以我希望我的BroadcastReciever
能够在不添加额外代码的情况下处理这两种类型的消息
(是的,我知道我可以为不同的BroadcastReciever
行动提供不同的intent-filter
,但我想实际上取消它,因为我知道可以做到,我很固执)
研究
我整天都在做研究。虽然我对数学和转换以及创建合适的算法非常糟糕,但我已经尝试自己编写。我查看了Stack topics on PDUs和Create PDU Android,但答案中的链接已断开。我甚至看了com.google.android.mms.pdu
源代码
到目前为止,我只能使用http://www.wrankl.de/JavaPC/SMSTools.html中的一些代码创建一个没有“始发地址”的PDU
PDU:
目的地:555消息:helloworld
"1100038155f50000aa0ae8329bfdbebfe56c32"
这显然无效......
旁注:
除了本地使用之外,我不打算对PDU做任何事情,我不想在我的代码中使用硬编码PDU,因为我没有重复使用PDU。
如果有什么我可以添加到我用来添加“始发地址”的代码中,那就可以了。或者有没有人知道我不知道的图书馆的信息?
由于
更新
尝试了
byte[] by =(byte[])(SmsMessage.getSubmitPdu("12345", "1234", "hello", false).encodedMessage);
给出了以下内容(以十六进制表示)
"0000100200000000000000000000000004010203040000000e000320ec400107102e8cbb366f00"
没有用
答案 0 :(得分:27)
也许这个代码段没有您想要的许多详细信息字段,但对于我的简单用途,它可以像其他短信一样调用通知。
private static void createFakeSms(Context context, String sender,
String body) {
byte[] pdu = null;
byte[] scBytes = PhoneNumberUtils
.networkPortionToCalledPartyBCD("0000000000");
byte[] senderBytes = PhoneNumberUtils
.networkPortionToCalledPartyBCD(sender);
int lsmcs = scBytes.length;
byte[] dateBytes = new byte[7];
Calendar calendar = new GregorianCalendar();
dateBytes[0] = reverseByte((byte) (calendar.get(Calendar.YEAR)));
dateBytes[1] = reverseByte((byte) (calendar.get(Calendar.MONTH) + 1));
dateBytes[2] = reverseByte((byte) (calendar.get(Calendar.DAY_OF_MONTH)));
dateBytes[3] = reverseByte((byte) (calendar.get(Calendar.HOUR_OF_DAY)));
dateBytes[4] = reverseByte((byte) (calendar.get(Calendar.MINUTE)));
dateBytes[5] = reverseByte((byte) (calendar.get(Calendar.SECOND)));
dateBytes[6] = reverseByte((byte) ((calendar.get(Calendar.ZONE_OFFSET) + calendar
.get(Calendar.DST_OFFSET)) / (60 * 1000 * 15)));
try {
ByteArrayOutputStream bo = new ByteArrayOutputStream();
bo.write(lsmcs);
bo.write(scBytes);
bo.write(0x04);
bo.write((byte) sender.length());
bo.write(senderBytes);
bo.write(0x00);
bo.write(0x00); // encoding: 0 for default 7bit
bo.write(dateBytes);
try {
String sReflectedClassName = "com.android.internal.telephony.GsmAlphabet";
Class cReflectedNFCExtras = Class.forName(sReflectedClassName);
Method stringToGsm7BitPacked = cReflectedNFCExtras.getMethod(
"stringToGsm7BitPacked", new Class[] { String.class });
stringToGsm7BitPacked.setAccessible(true);
byte[] bodybytes = (byte[]) stringToGsm7BitPacked.invoke(null,
body);
bo.write(bodybytes);
} catch (Exception e) {
}
pdu = bo.toByteArray();
} catch (IOException e) {
}
Intent intent = new Intent();
intent.setClassName("com.android.mms",
"com.android.mms.transaction.SmsReceiverService");
intent.setAction("android.provider.Telephony.SMS_RECEIVED");
intent.putExtra("pdus", new Object[] { pdu });
intent.putExtra("format", "3gpp");
context.startService(intent);
}
private static byte reverseByte(byte b) {
return (byte) ((b & 0xF0) >> 4 | (b & 0x0F) << 4);
}
希望你能找到有用的东西
更新:
public static final SmsMessage[] getMessagesFromIntent(
Intent intent) {
Object[] messages = (Object[]) intent.getSerializableExtra("pdus");
byte[][] pduObjs = new byte[messages.length][];
for (int i = 0; i < messages.length; i++) {
pduObjs[i] = (byte[]) messages[i];
}
byte[][] pdus = new byte[pduObjs.length][];
int pduCount = pdus.length;
SmsMessage[] msgs = new SmsMessage[pduCount];
for (int i = 0; i < pduCount; i++) {
pdus[i] = pduObjs[i];
msgs[i] = SmsMessage.createFromPdu(pdus[i]);
}
return msgs;
}
答案 1 :(得分:1)
自从我完成任何直接的PDU争吵以来,这已经很长时间了,但是当我这么做时,我很快放弃并使用了SMSLib:它已经完美地用于通过诺基亚手机发送的PDU实用程序(通过一系列链接)。我的假设(可能是错误的)是它们也适用于Android,假设接口实际上符合规范。
答案 2 :(得分:0)
请检查此code in console.c。这是android模拟器创建pdu和RIL.java的地方,其中CMT消息被转换为SmsMessage。您可以使用SmsMessage.getPdu
来获取pdu。但是SmsMessage.newFromCmt
看起来像是一个内部的api。所以它可能不可靠。
此外,这仅适用于Gsm,cdma具有完全不同的代码流,并且由于RIL.java和调制解调器完全是制造商特定的,因此这可能仅适用于模拟器。
通常GSM代码在Android上更可靠,所以不妨在gsm手机上工作。试一试。