frp部署
- 下载frp
- 配置frps服务器
- frpc配置
- 将frpc部署进项目
- 两个不同ip的主机连接frps
下载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; // 服务器监听的端口// 创建socketsocket_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主机一致:
运行后:
新人创作不易,你的点赞和关注都是对我莫大的鼓励,再次感谢您的观看。