4G短信猫发送中文短信(Python)
4G短信猫发送中文短信的方式可以使用 TEXT 模式或者 PDU 模式。
1.TEXT模式
在TEXT模式下发送中文短信的指令序列:
AT+CSCS="UCS2"
AT+CSMP=17,167,0,8
AT+CMGF=1
AT+CMGS="<手机号码>"
><短信内容的 UNICODE 编码>
<手机号码> - 必须是手机号码每一位的ASCII码值16进制表示的字符串,如:15148078817 是
00310035003100340038003000370038003800310037
<短信内容的 UNICODE 编码> - 直接将短信内容转换为 UNICODE 编码,可以通过百度搜索“unicode编码转换”或者直接使用站长工具:
https://tool.chinaz.com/Tools/Unicode.aspx,如“测试短信”的 Unicode 编码:16D4B8BD577ED4FE1
也可以自己写一个简单的代码即可实现,我这里用 Python 语言来实现:
def string2unicode(raw_str):
result = ''
for c in raw_str:
result += '%04x' % ord(c)
return result
通过 Python 语言来实现发送短信,需要使用 pyserial 库来实现,首先,安装:
pip install pyserial
然后,编写发送短信的代码如下:
# coding: utf-8
import serial
import settings
import time
from logger import logger
def string2unicode(raw_str):
result = ''
for c in raw_str:
result += '%04x' % ord(c)
return result
def send(mobile: str, msg: str):
ser = serial.Serial(settings.PORTX, settings.BPS, timeout=0.5)
try:
if not ser.isOpen(): ser.open()
ser.write('AT\r\n'.encode('ascii'))
res = ser.readall()
logger.info(res)
ser.write('AT+CMGF=1\r\n'.encode())
res = ser.readall()
logger.info(res)
ser.write('AT+CSCS="UCS2"\r\n'.encode())
res = ser.readall()
logger.info(res)
ser.write('AT+CSMP=17,167,0,8\r\n'.encode())
res = ser.readall()
logger.info(res)
ser.write('AT+CMGS="{0}"\r\n'.format(string2unicode(mobile)).encode())
res = ser.readall()
logger.info(res)
ser.write(string2unicode(msg).encode() + b'\x1a')
res = ser.readall()
logger.info(res)
except Exception as ex:
logger.error(str(ex))
finally:
time.sleep(5)
res = ser.readall()
logger.info(res)
if ser.isOpen(): ser.close()
对于 Ctri + Z 结束符,在写入串口时将 Unicode 编码的末尾加上 b'\x1a' 即可。
调用 send 函数发送短信:
if __name__ == '__main__':
for i in range(0,1):
send('15148078818', '{0}.I look for what I miss I know not what it is 寻寻觅觅 冷冷清清,【2023-01-11】'.format(i))
2.PDU模式
PDU 编码规则,我们以例子来说明:
手机号码:15148078818
短信内容:测试短信
PDU 编码:
0011000b815141088718F80008a7086D4B8BD577ED4FE1
PDU 编码长度:22
例子 | 说明 |
0011000b81 | 前缀 |
5141088718F8 | 手机号码 |
0008a7 | 附加码 |
08 | 短信内容的 Unicode 编码字节数,十六进制表示 |
6D4B8BD577ED4FE1 | 短信内容“测试短信”的 Unicode 编码 |
前缀和附加码:直接使用即可,对于一般发送国内的中文短信都可以满足,具体编码规则可网上查找;
手机号码:末尾用 f 将号码串补齐偶数位,字符两两调换位置。
Unicode 编码字节数:短信内容的 Unicode 编码后的字符数量除以2即可,例子中 Unicode 编码“6D4B8BD577ED4FE1”的长度为16,除以2得8,所以,其值为 x08。
PDU 长度:全部 PDU 编码不含最开始的 00 计算出的字符数除以 2,例子中的长度为 22。
在PDU模式下发送中文短信的指令序列:
AT+CSCS="UCS2"
AT+CMGF=0
AT+CMGS=
><短信内容的 PDU 编码>
编写发送短信的代码如下:
# coding: utf-8
import serial
import settings
import time
from logger import logger
def string2unicode(raw_str):
result = ''
for c in raw_str:
result += '%04x' % ord(c)
return result
# 手机号码编码转换
def mobile2code(mobile: str):
phone = mobile
phone = phone[1:] if phone.startswith('+') else phone
phone = phone if len(phone) % 2 == 0 else phone + 'F'
lst = []
for i in range(0, len(phone), 2):
a, b = phone[i + 1], phone[i]
lst.append(a)
lst.append(b)
return ''.join(lst)
# PDU编码转换
def pducode(mobile: str, msg: str):
phone = mobile2code(mobile)
logger.info('Mobile: %s' % phone)
message = string2unicode(msg)
logger.info('Unicode: %s' % message)
msg_length = '%02x' % (len(message) // 2)
logger.info('Unicode size: %s' % msg_length)
info = '0011000b81' + phone + '0008a7' + msg_length + message
logger.info('Info: %s' % info)
logger.info('Info size: %d' % ((len(info)-2) // 2))
return info
# 发送
def send(mobile: str, msg: str):
info = pducode(mobile, msg)
ser = serial.Serial(settings.PORTX, settings.BPS, timeout=0.5)
try:
if not ser.isOpen(): ser.open()
ser.write('AT\r\n'.encode('ascii'))
res = ser.readall()
logger.info(res)
ser.write('AT+CMGF=0\r\n'.encode())
res = ser.readall()
logger.info(res)
ser.write('AT+CSCS="UCS2"\r\n'.encode())
res = ser.readall()
logger.info(res)
ser.write('AT+CMGS={0}\r\n'.format((len(info)-2) // 2).encode())
res = ser.readall()
logger.info(res)
ser.write(info.encode() + b'\x1a')
res = ser.readall()
logger.info(res)
except Exception as ex:
logger.error(str(ex))
finally:
time.sleep(5)
res = ser.readall()
logger.info(res)
if ser.isOpen(): ser.close()
调用 send 函数发送短信:
if __name__ == '__main__':
for i in range(0,1):
send('15148078818', '{0}.I look for what I miss I know not what it is 寻寻觅觅 冷冷清清,【2023-01-11】'.format(i))