华为云短信服务教你用C++实现Smgp协议

news/2025/1/16 11:13:09/文章来源:https://www.cnblogs.com/huaweiyun/p/18241540

本文分享自华为云社区《华为云短信服务教你用C++实现Smgp协议》,作者:张俭。

引言&协议概述

中国联合网络通信有限公司短消息网关系统接口协议(SGIP)是中国网通为实现短信业务而制定的一种通信协议,全称叫做Short Message Gateway Interface Protocol,用于在短消息网关(SMG)和服务提供商(SP)之间、短消息网关(SMG)和短消息网关(SMG)之间通信。

Perl的IO::Async模块提供了一套简洁的异步IO编程模型。

SGIP 协议基于客户端/服务端模型工作。由客户端(短信应用,如手机,应用程序等)先和短信网关(SMG Short Message Gateway)建立起 TCP 长连接,并使用 SGIP 命令与SMG进行交互,实现短信的发送和接收。在SGIP协议中,无需同步等待响应就可以发送下一个指令,实现者可以根据自己的需要,实现同步、异步两种消息传输模式,满足不同场景下的性能要求。

时序图

连接成功,发送短信

 
 
 

连接成功,从SMGW接收到短信

 
 
 

协议帧介绍

image.png

SGIP Header

  • Message Length:长度为4字节,整个PDU的长度,包括Header和Body。
  • Command ID:长度为4字节,用于标识PDU的类型(例如,Login、Submit等)。
  • Sequence Number:长度为8字节,序列号,用来匹配请求和响应。

使用C++实现SMGP协议栈里的建立连接

├── CMakeLists.txt
├── examples
│   └── smgp_client_login_example.cpp
└── include└── sgipcpp├── BoundAtomic.h├── Client.h├── Protocol.h└── impl├── BoundAtomic.cpp├── Client.cpp└── Protocol.cpp

CMakeLists.txt:用来生成Makefile和编译项目

examples:存放示例代码
  • smgp_client_login_example.cpp:存放Smgp的login样例
include/sgipcpp:包含所有的C++头文件和实现文件
  • BoundAtomic.h:递增工具类,用来生成SequenceId
  • Client.h:Smgp定义,负责与Smgp服务进行通信,例如建立连接、发送短信等
  • Protocol.h:存放PDU,编解码等
  • impl/BoundAtomic.cpp:BoundAtomic类的实现
  • impl/Client.cpp:Client类的实现
  • impl/Protocol.cpp:Protocol中相关函数的实现

实现SequenceId递增

SequenceId是从1到0x7FFFFFFF的值,使用**BoundAtomic**类实现递增:

头文件

#ifndef BOUNDATOMIC_H
#define BOUNDATOMIC_H#include <atomic>
#include <cassert>class BoundAtomic {
public:BoundAtomic(int min, int max);int next_val();private:int min_;int max_;std::atomic<int> integer_;
};#endif //BOUNDATOMIC_H

内容

#include "sgipcpp/BoundAtomic.h"BoundAtomic::BoundAtomic(int min, int max) : min_(min), max_(max), integer_(min) {assert(min <= max);
}int BoundAtomic::next_val() {int current = integer_.load();int next;do {next = current >= max_ ? min_ : current + 1;} while (!integer_.compare_exchange_strong(current, next));return next;
}

实现SMGP PDU以及编解码函数

在**Protocol.h**中定义SMGP PDU以及编解码函数:

头文件

#ifndef PROTOCOL_H
#define PROTOCOL_H#include <cstdint>
#include <vector>constexpr uint32_t SGIP_BIND = 0x00000001;
constexpr uint32_t SGIP_BIND_RESP = 0x80000001;
constexpr uint32_t SGIP_UNBIND = 0x00000002;
constexpr uint32_t SGIP_UNBIND_RESP = 0x80000002;
constexpr uint32_t SGIP_SUBMIT = 0x00000003;
constexpr uint32_t SGIP_SUBMIT_RESP = 0x80000003;
constexpr uint32_t SGIP_DELIVER = 0x00000004;
constexpr uint32_t SGIP_DELIVER_RESP = 0x80000004;
constexpr uint32_t SGIP_REPORT = 0x00000005;
constexpr uint32_t SGIP_REPORT_RESP = 0x80000005;
constexpr uint32_t SGIP_ADDSP = 0x00000006;
constexpr uint32_t SGIP_ADDSP_RESP = 0x80000006;
constexpr uint32_t SGIP_MODIFYSP = 0x00000007;
constexpr uint32_t SGIP_MODIFYSP_RESP = 0x80000007;
constexpr uint32_t SGIP_DELETESP = 0x00000008;
constexpr uint32_t SGIP_DELETESP_RESP = 0x80000008;
constexpr uint32_t SGIP_QUERYROUTE = 0x00000009;
constexpr uint32_t SGIP_QUERYROUTE_RESP = 0x80000009;
constexpr uint32_t SGIP_ADDTELESEG = 0x0000000A;
constexpr uint32_t SGIP_ADDTELESEG_RESP = 0x8000000A;
constexpr uint32_t SGIP_MODIFYTELESEG = 0x0000000B;
constexpr uint32_t SGIP_MODIFYTELESEG_RESP = 0x8000000B;
constexpr uint32_t SGIP_DELETETELESEG = 0x0000000C;
constexpr uint32_t SGIP_DELETETELESEG_RESP = 0x8000000C;
constexpr uint32_t SGIP_ADDSMG = 0x0000000D;
constexpr uint32_t SGIP_ADDSMG_RESP = 0x8000000D;
constexpr uint32_t SGIP_MODIFYSMG = 0x0000000E;
constexpr uint32_t SGIP_MODIFYSMG_RESP = 0x8000000E;
constexpr uint32_t SGIP_DELETESMG = 0x0000000F;
constexpr uint32_t SGIP_DELETESMG_RESP = 0x8000000F;
constexpr uint32_t SGIP_CHECKUSER = 0x00000010;
constexpr uint32_t SGIP_CHECKUSER_RESP = 0x80000010;
constexpr uint32_t SGIP_USERRPT = 0x00000011;
constexpr uint32_t SGIP_USERRPT_RESP = 0x80000011;
constexpr uint32_t SGIP_TRACE = 0x00001000;
constexpr uint32_t SGIP_TRACE_RESP = 0x80001000;struct Header {uint32_t total_length;uint32_t command_id;uint64_t sequence_number;
};struct Bind {char login_type;char login_name[16];char login_passwd[16];char reserve[8];
};struct BindResp {char result;char reserve[8];
};struct Pdu {Header header;union {Bind bind;BindResp bind_resp;};
};size_t lengthBind();
std::vector<uint8_t> encodePdu(const Pdu& pdu);
Pdu decodePdu(const std::vector<uint8_t>& buffer);#endif //PROTOCOL_H

内容

#include "sgipcpp/Protocol.h"
#include <cstring>
#include <ostream>
#include <stdexcept>
#include <sys/_endian.h>size_t lengthBind(const Bind& bind) {return 1 + 16 + 16 + 8;
}void encodeBind(const Bind& bind, std::vector<uint8_t>& buffer) {size_t offset = 16;buffer[offset++] = bind.login_type;std::memcpy(buffer.data() + offset, bind.login_name, 16);offset += 16;std::memcpy(buffer.data() + offset, bind.login_passwd, 16);offset += 16;std::memcpy(buffer.data() + offset, bind.reserve, 8);
}BindResp decodeBindResp(const std::vector<uint8_t>& buffer) {BindResp bindResp;size_t offset = 0;offset += sizeof(uint32_t);offset += sizeof(uint32_t);bindResp.result = buffer[offset++];std::memcpy(bindResp.reserve, buffer.data() + offset, sizeof(bindResp.reserve));return bindResp;
}std::vector<uint8_t> encodePdu(const Pdu& pdu) {size_t body_length;switch (pdu.header.command_id) {case SGIP_BIND:body_length = lengthBind(pdu.bind);break;default:throw std::runtime_error("Unsupported command ID for encoding");}std::vector<uint8_t> buffer(body_length + 16);uint32_t total_length = htonl(body_length + 16);std::memcpy(buffer.data(), &total_length, 4);uint32_t command_id = htonl(pdu.header.command_id);std::memcpy(buffer.data() + 4, &command_id, 4);uint32_t sequence_number = htonl(pdu.header.sequence_number);std::memcpy(buffer.data() + 8, &sequence_number, 8);switch (pdu.header.command_id) {case SGIP_BIND:encodeBind(pdu.bind, buffer);break;default:throw std::runtime_error("Unsupported command ID for encoding");}return buffer;
}Pdu decodePdu(const std::vector<uint8_t>& buffer) {Pdu pdu;uint32_t command_id;std::memcpy(&command_id, buffer.data(), 4);pdu.header.command_id = ntohl(command_id);uint64_t sequence_number;std::memcpy(&sequence_number, buffer.data() + 8, 8);pdu.header.sequence_number = ntohl(sequence_number);switch (pdu.header.command_id) {case SGIP_BIND_RESP:pdu.bind_resp = decodeBindResp(buffer);break;default:throw std::runtime_error("Unsupported command ID for decoding");}return pdu;
}

实现客户端和登录方法

在**Client**中实现客户端和登录方法:

头文件

#ifndef CLIENT_H
#define CLIENT_H#include "BoundAtomic.h"
#include "Protocol.h"
#include "asio.hpp"
#include <string>class Client {
public:Client(const std::string& host, uint16_t port);~Client();void connect();BindResp bind(const Bind& bind_request);void close();private:std::string host_;uint16_t port_;asio::io_context io_context_;asio::ip::tcp::socket socket_;BoundAtomic* sequence_number_;void send(const std::vector<uint8_t>& data);std::vector<uint8_t> receive(size_t length);
};#endif //CLIENT_H

内容

#include "sgipcpp/Client.h"
#include <iostream>Client::Client(const std::string& host, uint16_t port): host_(host), port_(port), socket_(io_context_) {sequence_number_ = new BoundAtomic(1, 0x7FFFFFFF);
}Client::~Client() {close();delete sequence_number_;
}void Client::connect() {asio::ip::tcp::resolver resolver(io_context_);asio::connect(socket_, resolver.resolve(host_, std::to_string(port_)));
}BindResp Client::bind(const Bind& bind_request) {Pdu pdu;pdu.header.total_length = sizeof(Bind) + sizeof(Header);pdu.header.command_id = SGIP_BIND;pdu.header.sequence_number = sequence_number_->next_val();pdu.bind = bind_request;send(encodePdu(pdu));auto length_data = receive(4);uint32_t total_length = ntohl(*reinterpret_cast<uint32_t*>(length_data.data()));auto resp_data = receive(total_length - 4);Pdu resp_pdu = decodePdu(resp_data);return resp_pdu.bind_resp;
}void Client::close() {socket_.close();
}void Client::send(const std::vector<uint8_t>& data) {asio::write(socket_, asio::buffer(data));
}std::vector<uint8_t> Client::receive(size_t length) {std::vector<uint8_t> buffer(length);asio::read(socket_, asio::buffer(buffer));return buffer;
}

运行example,验证连接成功

#include "sgipcpp/Client.h"
#include <iostream>int main() {try {Client client("127.0.0.1", 8801);client.connect();std::cout << "Connected to the server." << std::endl;Bind bindRequest;bindRequest.login_type = 1;std::string login_name = "1234567890123456";std::string login_password = "1234567890123456";std::string reserve = "12345678";std::copy(login_name.begin(), login_name.end(), bindRequest.login_name);std::copy(login_password.begin(), login_password.end(), bindRequest.login_passwd);std::copy(reserve.begin(), reserve.end(), bindRequest.reserve);BindResp response = client.bind(bindRequest);if (response.result == 0) {std::cout << "Login successful." << std::endl;} else {std::cout << "Login failed with result code: " << static_cast<int>(response.result) << std::endl;}client.close();std::cout << "Connection closed." << std::endl;} catch (const std::exception& e) {std::cerr << "Error: " << e.what() << std::endl;}return 0;
}

image.png

相关开源项目

  • netty-codec-sms 存放各种SMS协议(如cmpp、sgip、smpp)的netty编解码器
  • sms-client-java 存放各种SMS协议的Java客户端
  • sms-server-java 存放各种SMS协议的Java服务端
  • cmpp-python cmpp协议的python实现
  • cngp-zig cmpp协议的python实现
  • sgip-cpp sgip协议的cpp实现
  • smgp-perl smgp协议的perl实现
  • smpp-rust smpp协议的rust实现

总结

本文简单对SGIP协议进行了介绍,并尝试用C++实现协议栈,但实际商用发送短信往往更加复杂,面临诸如流控、运营商对接、传输层安全等问题,可以选择华为云消息&短信(Message & SMS)服务通过HTTP协议接入,华为云短信服务是华为云携手全球多家优质运营商和渠道,为企业用户提供的通信服务。企业调用API或使用群发助手,即可使用验证码、通知短信服务。

点击关注,第一时间了解华为云新鲜技术~

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/722934.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

老生常谈!程序员为什么要阅读源代码?

面试造航母,入职拧螺丝。相信大家对这句话的精髓都深有体会,大家好,我是码农先森。 阅读源码这是一个老生常谈的话题了,但又是很多人想做又没有付出行动的事情。前段时间我研究了 Swoole 的源代码,并且输出了系列的源码分析文章「感兴趣的朋友可以翻阅以前的文章」。虽然这…

新增汇率无法保存,提示存在生效期间重叠的记录

金蝶提示信息有误,程序中考虑了原币和目标币相反的情况,如下图。

零一科技Yi-VL 6B视觉大模型环境搭建推理

​引子 最近看新闻博客说零一科技的Yi-VL-Plus视觉大模型效果很不错,那就想着尝尝鲜。这是第四篇关于视觉大模型的博客。之前有写过一篇零一科技的chat大模型零一科技Yi-34B Chat大模型环境搭建&推理_python部署大模型yi-34b-chat-CSDN博客,感兴趣的童鞋可以移步。我个人…

深度体验与测评openGauss 6.0.0新版本

openGauss 6.0.0版本在安装和使用方面都带来了很大的改进和优化。一站式交互安装功能极大地简化了安装流程,降低了用户的学习成本;性能优化和中文日志支持功能则进一步提升了数据库的稳定性和易用性。本文分享自华为云社区《openGauss 6.0.0新版本安装测评》,作者:马顺华。…

【IDEA 必备插件之一】这样注入 Bean 才爽

今天我们来介绍一款免费的IDEA生产力插件,它叫Bean Assistant。我们可以在插件市场搜索Bean Assistant来安装它。前言 不知道你们在平时的工作场景中是否经常遇到以下的情况。在一个方法中,需要调用某个实例接口的方法。我们经常会先在该类里面先注入这个接口的实例 Bean,然后…

文件系统(六):一文看懂linux ext4文件系统工作原理

liwen01 2024.06.09 前言 Linux系统中的ext2、ext3、ext4 文件系统,它们都有很强的向后和向前兼容性,可以在数据不丢失的情况下进行文件系统的升级。目前ext4是一个相对较成熟、稳定且高效的文件系统,适用于绝大部分规模和需求的Linux环境。 ext4它突出的特点有:数据分段管…

【译】Visual Studio 17.10 发布了新版扩展管理器

从 Visual Studio 17.10 开始提供新的扩展管理器作为默认预览功能。我们已将基本功能简化为现代风格 UI,以帮助您发现新的扩展并管理已安装的扩展。我们将更新的扩展管理器带给所有用户!在过去的一年里,我们已经将更新后的扩展管理器作为可选的预览功能提供,并一直期待您的…

手机上玩 PC 游戏的开源项目「GitHub 热点速览」

上周国产 3A 大作《黑神话:悟空》开启预售,同时公布游戏将于北京时间 2024.8.20 正式上线。这是一款由「游戏科学」开发的西游题材单机动作角色扮演游戏,它采用「虚幻引擎5」制作。该引擎并不是完全开源的,但它提供了部分源代码的访问权限。具体来说,就是 **GitHub 账号必…

Python 数据类型

Python 数据类型 Python 的数据类型大致可以分为两大类:基本数据类型和容器数据类型。基本数据类型通常指的是单一、不可分割的数据对象,而容器数据类型则用于存储多个数据对象的集合。如下图所示:基本数据类型 整型(int) 整型数据用于表示整数。Python 中的整型变量没有固…

[转帖]探索fio参数如何选择以及全方位对比HDD和SSD性能

文章目录 1. 磁盘I/O性能指标1.1 性能指标1.2 I/O 观测1.2.1 磁盘I/O 观测1.2.2 进程I/O观测 2. Fio 性能测试2.1 环境准备2.2 测试维度选择2.3 测试2.3.1 optane ssd和nvme ssd性能测试2.3.2 aep性能测试(intel persistent memory) 真正测试之前 我们需要清楚 评判磁盘I/O性能…

3. 使用Mybatis完成CRUD

前置工作准备创建Maven项目 , 引入依赖(mybatis依赖 ,mysql 驱动依赖 ,junit依赖 ,logback 依赖) 将xml文件放到类的根路径下 提供com.north.mybatis.utils.SqlSessionUtil工具类 创建测试用例:com.north.mybatis.CarMapperTest‍ 补充知识:什么是CRUD C: Create增 R: Re…

[转帖]JDK/Java 17 GA,新增「Free Java License」

https://zhuanlan.zhihu.com/p/410556607 按照发布规划,JDK/Java 17 属于长期支持版本 (LTS),将会获得 8 年的技术支持,直至 2029 年 9 年。JDK/Java 17 总共包含 14 个 JEP,具体如下:Restore Always-Strict Floating-Point Semantics 恢复始终执行严格模式 (Always-Str…