本文分享自华为云社区《华为云短信服务教你用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接收到短信

 
 
 

协议帧介绍

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;
}

相关开源项目

总结

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

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

华为云短信服务教你用C++实现Smgp协议的更多相关文章

  1. php 阿里云短信服务及阿里大鱼实现短信验证码的发送

    一:使用阿里云的短信服务 ① 申请短信签名 ②申请短信模板 ③创建Access Key,获取AccessKeyId 与 AccessKeySecret.(为了安全起见,这里建议使用子用户的Access ...

  2. 阿里云短信服务bug

    接入阿里云短信服务,在springboot中写测试方法,执行到 IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou ...

  3. 阿里云短信服务调用例子-Python

    阿里云短信服务调用例子 阿里云官方文档https://helpcdn.aliyun.com/document_detail/101893.html 首先需要安装阿里云PythonSDK(下面是pyth ...

  4. ThinkPHP3.2.3框架下接入阿里云短信服务接口实现:注册登录

    首先介绍下短信注册登录流程: 注册页面点击获取手机号验证码按钮,用jquery的click事件POST或GET方法把手机号发送到后台控制器: 后台控制器创建函数,收到手机号后生成随机码,例如:6位的随 ...

  5. 移动端获取短信验证码java实现——阿里云短信服务

    需求:移动端输入手机号,获取验证码.点击登录,验证验证码是否输入错误.是否超时等情况,一旦校验通过,将用户数据保存到数据中(业务逻辑). 前提:注册阿里用户,开通短信服务,申请key.秘钥.签名.短信 ...

  6. 浏览器端获取短信验证码java实现——阿里云短信服务

    需求:浏览器端输入手机号,获取验证码.点击登录,验证验证码是否输入错误.是否超时等情况,一旦校验通过,将用户数据保存到数据中(业务逻辑). 前提:注册阿里用户,开通短信服务,申请key.秘钥.签名.短 ...

  7. flask+阿里云短信服务实现注册发送手机验证码

    效果图: 该效果主要讲解实现通过调用阿里云的SDK实现发送注册验证码短信(阿里云短信付费使用) 购买阿里云短信服务 购买链接:https://www.aliyun.com/product/sms 1. ...

  8. springboot 使用阿里云短信服务发送验证码

    一.申请阿里云短信服务 1.申请签名 2.申请模板 3.创建accesskey(鼠标悬停在右上角头像) 二.代码实现 1.springboot引入maven依赖 <dependency> ...

  9. 使用springboot集成腾讯云短信服务,解决配置文件读取乱码问题

    springboot集成腾讯云短信服务: (1)导入依赖 <dependency> <groupId>org.springframework.boot</groupId& ...

  10. Zabbix 3.4.3 使用阿里云短信服务进行报警

    目录 一.阿里云短信服务 1.1.首先开通阿里云短信服务 1.2 创建签名 1.3 创建短信模板 1.4 创建发送脚本 二.Zabbix Web 配置 2.1 增加 Media types 2.2 给 ...

随机推荐

  1. 力扣283(java)-移动零(简单)

    题目: 给定一个数组 nums,编写一个函数将所有 0 移动到数组的末尾,同时保持非零元素的相对顺序. 请注意 ,必须在不复制数组的情况下原地对数组进行操作. 示例 1: 输入: nums = [0, ...

  2. 力扣609(java&python)-在系统中查找重复文件(中等)

    给你一个目录信息列表 paths ,包括目录路径,以及该目录中的所有文件及其内容,请你按路径返回文件系统中的所有重复文件.答案可按 任意顺序 返回. 一组重复的文件至少包括 两个 具有完全相同内容的文 ...

  3. 技术解读 | 智能开放搜索CTR预估模型

    ​简介:本文介绍开放搜索CTR预估模型在个性化排序中的应用与实践 如何评价搜索排序效果? 搜索是用户触达信息最简单直接的方式,是APP.网页必备的功能.如何评价并提升搜索效果一直是搜索领域的常见问题. ...

  4. 当设计模式遇上 Hooks

    ​简介: 数据结构与设计模式能够指导我们在开发复杂系统中寻得一条清晰的道路,既然都说 Hooks 难以维护,那就尝试让「神」来拯救这混乱的局面.对于「设计模式是否有助于我们写出更优雅的 Hooks 」 ...

  5. [GPT] Vue 的 methods 中使用了 addEventListener,如何在 addEventListener 的匿名函数参数中访问 Vue data 变量

      在 Vue 的 methods 方法中使用 addEventListener时,你可以使用 箭头函数 来访问 Vue 实例的数据. 箭头函数不会创建自己的作用域,而是继承父级作用域的上下文.以下是 ...

  6. dotnet 教你写一个可以搞炸本机所有 WCF 应用的程序方法

    作为团队里面挖掘机出身的我,怎么能不多挖一些坑好将小伙伴们都埋进去呢.本文来告诉大家一个有趣且简单的方法,此方法可以将本机的 WCF 玩坏,不敢说真的搞炸本机所有 WCF 应用,但搞炸大部分基于 WC ...

  7. Oracle、达梦:数据库大小写不敏感,但是又要区分大小写敏感(默认敏感)

    一. 艹,这个需求就很操蛋. 实现 SELECT * FROM T1 WHERE REGEXP_LIKE(field, '.*value.*', 'c'); 在 Oracle 数据库中使用 REGEX ...

  8. CF1097C Yuhao and a Parenthesis

    CF1097C Yuhao and a Parenthesis stl 乱搞做法,感觉比正解更直接. 每个字符串内部能匹配的尽可能匹配. 匹配完成后,检验剩余序列是否只含有 ( 或只含有 ) 或为空, ...

  9. SAP UI5 官方教程学习记录

    最近有闲跟着官方的Get Started教程学习了UI5,记录一下自己学习中遇到的几个问题. 本文链接:https://www.cnblogs.com/hhelibeb/p/17835722.html ...

  10. width:100%与width:auto区别

    小知识 width:100%与width:auto区别 width:100% : 子元素的 content 撑满父元素的content,如果子元素还有 padding.border等属性,或者是在父元 ...