下载frp
目前下载的是frp5.9版本,(6.0使用的时候有问题)。将压缩包上传服务器并解压:
进入frp5.9的目录:
配置frps服务器
frps.toml是frps的配置文件,进行配置:
- bindPort : 用来和内网主机客户端连接的端口。
- auth.method = “token” : 服务端连接身份认证,默认token。
- auth.token = “test123” : 服务端token密码,auth.method和auth.token要求frpc保持一致
- 下面4个都是frps在web上的一个仪表盘,可以可视化的显示frps和frpc的连接情况
- webServer.addr = “0.0.0.0” : 访问web的ip
- webServer.port = 7600 : 访问web的端口
- webServer.user = “ambimaster” : 登录的用户名
webServer.password = “ab123456” : 登录的密码
配置完以后运行:
web上也登录成功:
frpc配置
将frp5.9放在项目下:
编写frpc.toml为:
- serverAddr : frps服务器ip
- serverPort: frps端口,也就是frps.toml的bindPort
- name : 可以有不同的ip通过bindPort和frps连接,但是name不能相同,不然会报错。
- localPort : 主机监听的端口,外网主机发出数据,内网主机通过监听localPort获取数据。
- remotePort : 外网主机通过向frps中的remotePort端口号发送数据 ,frpc通过localPort获得数据。
在frpc文件目录下执行命令:
frps和frpc成功连接。
将frpc部署进项目
下面代码主要测试为主,后面在根据需求具体修改
将frpc文件夹放在项目中
void MainWindow::frpc()
{
std::wstring batFilePath = L"..\\frp_5.9\\frpc.exe -c ..\\frp_5.9\\frpc.toml";
STARTUPINFO si;
PROCESS_INFORMATION pi;
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
ZeroMemory(&pi, sizeof(pi));
std::thread t1([&]()
{
// 启动frpc进程,不显示窗口
if (!CreateProcessW(NULL, // 不使用模块名
&batFilePath[0], // 命令行
NULL, // 默认进程安全性
NULL, // 默认线程安全性
FALSE, // 句柄不继承
CREATE_NO_WINDOW, // 不显示窗口
NULL, // 使用父进程的环境块
NULL, // 使用父进程的驱动器目录
&si, // 指针到STARTUPINFO结构体
&pi) // 指针到PROCESS_INFORMATION结构体
)
{
qDebug() << "CreateProcess failed (" << GetLastError() << ").";
return;
}
});
t1.join();
frpcProcess = pi.hProcess;
QTcpServer* tcpSocket = new QTcpServer(this);
connect(tcpSocket, &QTcpServer::newConnection, this, [=]()
{
QTcpSocket* tcpClientSocket = tcpSocket->nextPendingConnection();
connect(tcpClientSocket, &QTcpSocket::readyRead, this, [=]()
{
QByteArray data = tcpClientSocket->readAll();
qDebug() << "=++++++++++++++++++++++++++++++++++++++++read:>" << data;
});
tcpClientSocket->write("你好 你好 你好 你好 你好");
});
if (!tcpSocket->listen(QHostAddress::Any, 22)) // 监听22端口
{
qCritical() << "Failed to start server: " << tcpSocket->errorString();
return;
}
}
上面代码实现了运行frpc命令,使得启动项目就可使frpc连接frps,并且让其后台运行。
之后监听端口22,并将收到的数据打印出来,且回复一段消息。
使用其他ip的机器编写如下代码:
#include <iostream>
#include <string>
#include <cstring> // 用于memset
#include <sys/socket.h>
#include <netdb.h> // gethostbyname
#include <arpa/inet.h> // inet_addr, htons
#include <unistd.h>
#define MAX_BUFFER_SIZE 4096 // 最大缓冲区大小
int main() {
int socket_desc, port, client_len;
struct sockaddr_in server_addr;
char buffer[MAX_BUFFER_SIZE];
std::string message;
// 服务器IP地址和端口号
const char *server_ip = "8.155.161.216"; // 或者使用域名
port = 8100; // 服务器监听的端口
// 创建socket
socket_desc = socket(AF_INET, SOCK_STREAM, 0);
if (socket_desc == -1) {
std::cerr << "Could not create socket" << std::endl;
return -1;
}
// 配置服务器地址结构体
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port); // 将主机字节序转换为网络字节序
if (inet_pton(AF_INET, server_ip, &server_addr.sin_addr) <= 0) {
std::cerr << "Invalid address/Address not supported" << std::endl;
return -1;
}
// 连接到服务器
if (connect(socket_desc, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
perror("connect failed. Error");
return -1;
}
std::cout << "Connected to the server" << std::endl;
while(1)
{
sleep(1);
// 发送数据到服务器
message = "Hello, Server!";
if (send(socket_desc, message.c_str(), message.length(), 0) < 0) {
perror("Send failed");
return -1;
}
std::cout << "Message sent to server: " << message << std::endl;
// 接收服务器响应
memset(buffer, 0, MAX_BUFFER_SIZE); // 清空缓冲区
std::cout << "Waiting for server response..." << std::endl;
int bytes_received = recv(socket_desc, buffer, MAX_BUFFER_SIZE - 1, 0);
if (bytes_received <= 0) {
std::cerr << "Receive failed or connection closed by server" << std::endl;
} else {
std::cout << "Server response: " << buffer << std::endl;
}
}
return 0;
}
主要为了测试,向frps服务器ip8.155.161.216
, remotePort端口:8100
发送数据,并接受响应,将响应的数据回显到终端面板。
下面是测试:
运行项目:
服务端显示已连接:
web上:
运行外网主机上编写的代码:
可以收到内网主机的回复
两个不同ip的主机连接frps
此时让A主机frpc.toml里面的name为:test-tcp1
另外主机上的name为test-tcp2:
启动A主机的frpc:
启动另外主机的frpc:
结果:
两台主机已连接。
如果将其中一台主机的frpc.toml的name修改为:test-tcp1和A主机一致:
运行后:
新人创作不易,你的点赞和关注都是对我莫大的鼓励,再次感谢您的观看。