Tauri 提供界面 + 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令

Tauri 提供界面 + 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令

文章目录

  • Tauri 提供界面 + 使用 Rust 实现连接远程 Linux 服务器、发送文件、执行命令
  • 一、Tauri 概述
  • 二、界面预览
  • 三、代码参考
    • 1、main.rs
    • 2、App.vue
    • 3、Greet.vue
    • 4、依赖

一、Tauri 概述

Tauri 是一个用于构建跨平台桌面应用程序的工具和框架。它的目标是通过将 Web 技术与本地功能结合,使开发者能够以简单、高效的方式创建现代的桌面应用。

以下是 Tauri 的一些主要特点和概述:

  1. 跨平台支持:Tauri 允许你构建跨平台的桌面应用程序,它支持 Windows、macOS 和 Linux 等常见的操作系统。你可以使用一套代码库在多个平台上构建应用程序。
  2. 基于 Web 技术:Tauri 使用 Web 技术作为应用程序的前端开发语言。你可以使用 HTML、CSS 和 JavaScript(或其他 Web 前端框架)来构建应用程序的用户界面。
  3. 原生功能访问:Tauri 提供了访问原生功能的接口,让你可以从前端代码中直接调用本地操作系统的功能,如文件系统、网络、系统通知等。这样,你可以创建出与本地应用程序类似的功能和体验。
  4. 嵌入式 Web 渲染引擎:Tauri 使用嵌入式的 Web 渲染引擎(如 WebView 或 WebKitGTK)来渲染应用程序的界面。这使得应用程序可以直接在桌面环境中运行,而无需依赖外部的浏览器。
  5. 丰富的生态系统:Tauri 生态系统提供了许多有用的功能和库,如打包工具、插件系统、前端构建工具等,以便于应用程序的开发和部署。
  6. 灵活的扩展性:Tauri 允许你通过使用 Rust 和 JavaScript 进行扩展,从而实现更复杂的功能。你可以编写原生 Rust 代码来访问底层的系统功能,并使用 JavaScript 与前端代码进行交互。

总的来说,Tauri 提供了一个快速、简单的方式来开发跨平台的桌面应用程序。通过结合 Web 技术和原生功能,你可以创建出功能丰富、具有优秀用户体验的桌面应用。无论是构建独立的应用程序,还是将现有的 Web 应用转化为桌面应用,Tauri 都是一个强大的选择。

二、界面预览

image-20230709233228823

三、代码参考

1、main.rs

// Prevents additional console window on Windows in release, DO NOT REMOVE!!
#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")]
#![allow(unused_assignments)] // 禁用未使用赋值的警告use std::io::{Read, Write};
use std::path::Path;
use ssh2::Session;// Learn more about Tauri commands at https://tauri.app/v1/guides/features/command
#[tauri::command]
fn go(ip_with_port: &str, username: &str, password: &str, local_file_path: &str, target_file_path: &str, command: &str) {publish(ip_with_port, username, password, local_file_path, target_file_path, command)
}fn main() {tauri::Builder::default().invoke_handler(tauri::generate_handler![go]).run(tauri::generate_context!()).expect("error while running tauri application");
}const PROGRESS_UPDATE_INTERVAL: usize = 1024 * 1024; // 每传输 1MB 打印一次进度信息// 发布
fn publish(ip_with_port: &str, username: &str, password: &str, local_file_path: &str, target_file_path: &str, command: &str) {// 连接远程服务器let tcp = std::net::TcpStream::connect(ip_with_port).unwrap(); // 连接到远程服务器let mut sess = Session::new().unwrap(); // 创建一个新的会话sess.set_tcp_stream(tcp); // 设置会话的 TCP 流sess.handshake().unwrap(); // 进行握手sess.userauth_password(username, password).unwrap(); // 使用用户名和密码进行身份验证// 传输文件let file_size = get_file_size(local_file_path); // 获取文件大小let mut channel = sess.scp_send(Path::new(target_file_path), 0o644, file_size, None).unwrap(); // 创建一个新的 SCP 通道let mut file = std::fs::File::open(local_file_path).unwrap(); // 打开本地文件let mut buffer = Vec::new(); // 创建一个空的字节向量file.read_to_end(&mut buffer).unwrap(); // 读取文件内容let mut total_bytes_sent = 0; // 已发送的总字节数let mut total_mb = (file_size as f64) / (1024.0 * 1024.0); // 文件总大小(MB)total_mb = (total_mb * 100.0).round() / 100.0; // 保留2位小数let mut transferred_mb = 0.0; // 已传输的文件大小(MB)// 记录已发送文件的大小let mut bytes_sent = 0; // 已发送的字节数for (i, chunk) in buffer.chunks(PROGRESS_UPDATE_INTERVAL).enumerate() { // 循环发送文件内容// 循环发送,发完为止!while bytes_sent < chunk.len() {let result = channel.write(chunk).unwrap(); // 发送文件内容bytes_sent += result; // 更新已发送的字节数}total_bytes_sent += bytes_sent; // 更新已发送的总字节数bytes_sent = 0; // 重置已发送的字节数transferred_mb = (total_bytes_sent as f64) / (1024.0 * 1024.0); // 更新已传输的文件大小(MB)transferred_mb = (transferred_mb * 100.0).round() / 100.0; // 保留2位小数if (i + 1) * PROGRESS_UPDATE_INTERVAL <= buffer.len() { // 如果还有剩余的文件内容需要发送let progress = (total_bytes_sent as f64) / (file_size as f64) * 100.0; // 计算传输进度println!("进度: {:.2}% ({:.2} MB / {:.2} MB)", progress, transferred_mb, total_mb); // 打印传输进度信息} else { // 文件传输完毕println!("进度: 100% 文件传输完毕!");}}channel.send_eof().unwrap(); // 发送 EOF 标志// 执行远程命令let mut channel = sess.channel_session().unwrap(); // 创建一个新的会话通道channel.exec(command).unwrap(); // 执行命令let mut output = Vec::new(); // 创建一个空的字节向量channel.read_to_end(&mut output).unwrap(); // 读取命令输出println!("{}", String::from_utf8_lossy(&output)); // 打印命令输出
}// 获取文件大小
fn get_file_size(file_path: &str) -> u64 {std::fs::metadata(file_path) // 获取文件元数据.map(|metadata| metadata.len()) // 获取文件大小.unwrap_or(0) // 如果获取失败,则返回 0
}

2、App.vue

<script setup lang="ts">
import Greet from "./components/Greet.vue";
</script><template><div class="container"><h1>Spring Boot 程序发布工具!</h1><div class="tips">程序发布 = 连接 Linux + 发送文件 + 执行命令</div><Greet /></div>
</template><style scoped>
.tips {color: #666666;margin-bottom: 10px;
}
</style>

3、Greet.vue

<template><form class="row" @submit.prevent="go"><input v-model="ipWithPort" placeholder="IP地址:端口号"/><input v-model="username" placeholder="账号"/><input v-model="password" placeholder="密码" type="password"/><input v-model="filePath" placeholder="本地文件路径"/><input v-model="targetPath" placeholder="目标文件路径"/><input v-model="command" placeholder="命令"/><button type="submit">执行!</button></form><p>{{result}}</p>
</template><script setup lang="ts">
import {ref} from "vue";
import {invoke} from "@tauri-apps/api/tauri";// IP地址和端口号:ip:port
const ipWithPort = ref("222.222.222.222:22");
// 账号
const username = ref("root");
// 密码
const password = ref("root");
// 本地文件路径
const filePath = ref("C:\\\\Users\\\\Administrator\\\\Desktop\\\\app.jar");
// 目标文件路径
const targetPath = ref("/home/zibo/app.jar");
// 命令
const command = ref("pwd");
// 执行结果
const result = ref("未开始执行!");async function go() {result.value = "执行中...";await invoke("go", {ipWithPort: ipWithPort.value, username: username.value, password: password.value, localFilePath: filePath.value, targetFilePath: targetPath.value, command: command.value});result.value = "执行完毕!";
}
</script><style>
.row {display: flex;flex-direction: column;margin: 0 10px;
}.row input {margin-bottom: 10px;
}
</style>

4、依赖

Cargo.toml

# ssh2
ssh2 = "0.9.4"

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

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

相关文章

HummerRisk 配置 HTTPS访问

简介 HummerRisk 是开源的云原生安全平台&#xff0c;以非侵入的方式解决云原生的安全和治理问题。核心能力包括混合云的安全治理和云原生安全检测。 HTTPS&#xff08;Hypertext Transfer Protocol Secure&#xff09;是一种通过加密和身份验证来保护网络通信安全的协议。它…

软件设计模式与体系结构-软件体系-数据流风格软件体系结构

二、数据流风格软件体系结构 概念 数据流系统的各软件组件在无数据到达时处于休眠状态&#xff0c;当有数据到达时&#xff0c;该软件组件被激活&#xff0c;开始对数据进行处理。数据流风格软件体系结构将整个软件系统看作由一系列作用在连续数据集合上的“变换”组成。数据…

第N4周:使用Word2vec实现文本分类

目录 二、数据预处理1.加载数据2.构建词典3.生成数据批次和迭代器 二、模型构建1.搭建模型2.初始化模型3.定义训练与评估函数 三、训练模型1.拆分数据集并运行模型2.测试指定数据 &#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&…

测试服务器CPU情况

要查看服务器的CPU情况&#xff0c;你可以使用 TOP 命令结合一些选项来执行相应的测试top 命令&#xff1a;运行 top 命令可以实时监视系统的各个进程和 CPU 使用情况在 top 命令的输出中&#xff0c;有几个重要的指标参数可以帮助你了解系统的 CPU 使用情况和进程信息。以下是…

Mac电脑如何与 Windows 用户共享文件?

在 Mac 上设置文件共享 选取苹果菜单 >“系统偏好设置”&#xff0c;然后点按“共享”。 选择“文件共享”复选框&#xff0c;然后点按“选项”。 选择“使用 SMB 来共享文件和文件夹”复选框。 在“Windows 文件共享”列表中&#xff0c;选择要与 Windows 用户共享文件的…

JavaWeb_瑞吉外卖_项目优化Day11-Mysql主从复制, Nginx

JavaWeb_瑞吉外卖_项目优化Day11-Mysql主从复制, Nginx Mysql主从复制介绍配置主库Master从库Slave 读写分离案例背景Sharding-JDBC介绍入门案例 Nginx概述介绍下载安装目录结构 命令配置文件结构具体应用部署静态资源反向代理负载均衡 来源 Mysql主从复制 介绍 MySQL主从复制…

数据结构--由遍历序列构造二叉树

数据结构–由遍历序列构造二叉树 不同二叉树的中序遍历序列 中序遍历:中序遍历 左 \color{red}左 左子树、 根 \color{purple}根 根结点、中序遍历 右 \color{green}右 右子树 中序遍历序列:BDCAE 结论 : 一个中序遍历序列可能对应多种二叉树形态 \color{green}结论:一个中序…

[NISACTF 2022]babyserialize(pop链构造与脚本编写详细教学)

目录 一、理清pop链并进行标注 二、如何编写相关脚本 三、过滤与绕过 1、waf的绕过 2、preg_match的绕过 做这道题作为pop链的构造很典型&#xff0c;也很有意思&#xff0c;因为还存在一些其他东西。 打开链接&#xff0c;这种很多类的PHP代码多半是需要构造pop链 一、理…

stm32_<一文通>_cubemx_freertos

文章目录 前言一、任务调度1.1 延时1.1.1 相对延时1.1.2 绝对延时 1.2 挂起和恢复1.2.1 cmsis的挂起和恢复函数1.2.2 freertos的挂起和恢复函数 1.3 删除1.3.1 cmsis的删除任务函数1.3.2 freertos的删除任务函数 二、Freertos任务与中断三、消息队列3.1 写入和读取一个数据3.2 …

什么是数据一致性

什么是数据一致性 数据一致性这个单词在平常开发中&#xff0c;或者各种文章中都能经常看见&#xff0c;我们常常听见什么东西数据不一致了&#xff0c;造成了一定的损失&#xff0c;赶快修复一下。但是很多同学对一致性具体代表什么意思&#xff0c;他有什么作用依然不是很了解…

一文弄懂Java日志框架

文章目录 日志的概念日志门面JUL日志框架JUL架构入门案例日志的级别Logger之间的父子关系日志的配置文件日志原理解析 LOG4J日志框架Log4j入门Log4j组件LoggersAppendersLayouts Layout的格式Appender的输出自定义Logger JCL日志门面JCL入门JCL原理 SLF4J日志门面SLF4J入门绑定…

Redis优化

目录 一、Redis高可用 二、Redis持久化 1.RDB持久化 1.1触发条件 1.1.1手动触发 1.1.2自动触发 1.2其他自动触发机制 1.3执行流程 1.4启动时加载 2.AOF 持久化 2.1开启AOF 2.2执行流程 2.2.1命令追加(append) 2.2.2文件写入(write)和文件同步(sync) 2.2.3文件重…