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

news2024/12/23 11:12:21

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.coloradmin.cn/o/748457.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C语言动态获取设备的网络接口名称和状态以及对应的IP地址

一、目的 在实际项目中需要获取设备的IP地址然后通过广播的形式通知局域网内的其他设备。 二、介绍 方法一 通过ioctl方式获取SIOCGIFADDR信息 /** C Program to Get IP Address*/ #include <stdio.h> #include <string.h> #include <sys/types.h> #includ…

mfc120u.dll丢失修复,mfc120u.dll缺失的解决方法

MFC120u.dll缺失的原因 当系统中缺少或损坏了MFC120u.dll文件时&#xff0c;就会出现"MFC120u.dll缺失"的错误提示。造成MFC120u.dll缺失的原因可能有以下几种情况&#xff1a; 1.文件删除或损坏&#xff1a;MFC120u.dll文件可能因为误删除、病毒感染、硬盘故障等原…

pearsonr报错:计算结果为nan、warning .warn (stats.constantinputwarning (msg))

【1】两个都是ndarry 最后结果为nan&#xff1a; &#xff08;1&#xff09;数据类型转换&#xff1a;都转为一样的float32&#xff1a;依旧报错nan &#xff08;2&#xff09;进入函数内部debug&#xff1a; if (xx[0]).all() or (yy[0]).all() warning .warn (stats.consta…

13-接口、代码、命令如何测试?

可以通过工具&#xff0c;也可以通过代码来进行测试。 这里使用工具Postman。 1.针对接口进行测试 粘贴导入到Postman中&#xff1a; 也可直接复制URL&#xff1a; 粘贴到Postman中&#xff1a; 没有对应的参数&#xff0c;那么可以直接请求&#xff1a; 这个接口的GET、POST…

无法将“pip“识别为cmdlet、函数、脚本文件或可运行程序的名称。

出现问题如下&#xff1a; 出现问题原因&#xff1a; 没有添加pip对应的安装目录进入环境变量里面的系统变量。 解决方案&#xff1a; 1.确定python的安装路径 将python的路径添加到系统变量中 2.输入pip所在的安装路径&#xff1a; python路径\Lib\site-packages 3.添加…

PyTorch训练RNN, GRU, LSTM:手写数字识别

文章目录 pytorch 神经网络训练demoResult参考来源 pytorch 神经网络训练demo 数据集&#xff1a;MNIST 该数据集的内容是手写数字识别&#xff0c;其分为两部分&#xff0c;分别含有60000张训练图片和10000张测试图片 图片来源&#xff1a;https://tensornews.cn/mnist_intr…

eureka使用错误

错误 java.net.UnknownHostException: INVENTORYSERVICE 分析&解决&#xff1a; 这里的请求执行错误 但eureka可以找到服务 手动创建RestTemlate到容器中&#xff0c;未加LoadBalanced注解 加上注解后重试&#xff0c;成功

java详细显示try/catch块的异常类方法等信息

示例&#xff1a; Testpublic void testException(){try {double theorynumDouble Double.parseDouble(null);} catch (Exception e) {StackTraceElement[] stackTraceElements e.getStackTrace();StackTraceElement stackTraceElementFirst stackTraceElements[0];String c…

一文教你学会Linux数组

目录 &#x1f380;什么是数组&#xff1f; &#x1f380;数组优点 &#x1f380;数组缺点 &#x1f380;定义数组 &#x1f380;数组的取值 &#x1f380;一次取出数组所有的值 &#x1f380;数组长度&#xff1a; 即数组元素个数 &#x1f380;数组的截取&#xff…

【ASP.NET】医学实验室管理(LIS)系统源码

一、医学实验室LIS系统概况 LIS&#xff08;全称Laboratory Information Management System&#xff09;&#xff0c;是专为医院检验科设计的一套实验室信息管理系统&#xff0c;能将实验仪器与计算机组成网络&#xff0c;使病人样品登录、实验数据存取、报告审核、打印分发&am…

MySQL数据库的索引原理与慢SQL优化的5大原则

这篇文章主要介绍了MySQL数据库的索引原理与慢SQL优化的5大原则,包括&#xff1a;建立索引的原则&#xff0c;慢查询优化基本步骤&#xff0c;慢查询优化案例&#xff0c;explain使用&#xff0c;需要的朋友可以参考下 我们知道一般的应用系统&#xff0c;读写比例在10:1左右&…

计算机系统结构与操作系统实验三(2)-进入保护模式

&#x1f4cd;实验要求 从实模式到保护模式的转变&#xff1a; 在刚进入loader尚在实模式下时&#xff0c;在第2行显示&#xff1a;real-Zhangsan 在变为保护模式后&#xff0c;在第3行显示&#xff1a;protect-Zhangsan 加载gdt、将cr0的PE位置为1 &#x1f4cd;实验过程 老…

建议收藏 | 可视化ETL平台--Kettle

Kettle的介绍 学习目标 知道什么是ETL及Kettel是开源的ETL工具 了解kettle环境的安装流程 1.ETL介绍 ETL&#xff08;Extract-Transform-Load的缩写&#xff0c;即数据抽取、转换、装载的过程&#xff09;&#xff0c;对于企业或行业应用来说&#xff0c;我们经常会遇到各种…

redis可视化工具 Another Redis Desktop Manager

Redis API 官方文档&#xff1a;Redis 命令参考 — Redis 命令参考 Redis 可视化工具下载&#xff1a; Releases qishibo/AnotherRedisDesktopManager GitHub&#xff08;github&#xff09; 使用&#xff1a; 1、新建连接 点击 新建连接 创建新的连接 切换语言 地址&a…

前端Vue自定义商品订单星级评分 爱心评分组件

随着技术的发展&#xff0c;开发的复杂度也越来越高&#xff0c;传统开发方式将一个系统做成了整块应用&#xff0c;经常出现的情况就是一个小小的改动或者一个小功能的增加可能会引起整体逻辑的修改&#xff0c;造成牵一发而动全身。 通过组件化开发&#xff0c;可以有效实现…

数据分类分级的概念、方法、标准及行业实践

数据战略上升为国家战略&#xff0c;数据资产成为国家各行各业的核心资产。在数字化时代&#xff0c;数据分类分级成为数据资产管理的重要组成部分。 通过数据分类分级管理&#xff0c;可有效使用和保护数据&#xff0c;使数据更易于定位和检索&#xff0c;满足数据风险管理、…

领略大数据分析的魅力:迈向FineBI的世界

版权声明&#xff1a;本文为 小异常 原创文章&#xff0c;非商用自由转载-保持署名-注明出处&#xff0c;谢谢&#xff01; 本文网址&#xff1a;https://sunkuan.blog.csdn.net/article/details/131698171 文章目录 一、FineBI介绍&#xff08;6.0版本&#xff09;1、使用人群…

高并发的哲学原理(三)-- 基础设施并发:虚拟机与 Kubernetes(k8s)

上篇文章说到&#xff0c;Apache 无法处理海量用户的 TCP 连接&#xff0c;那要是由于宇宙时空所限&#xff0c;你的系统就是无法离开 Apache&#xff0c;该怎么承接高并发呢&#xff1f;有办法&#xff1a;既然单机不行&#xff0c;那就把单机虚拟化成多个 Linux 机器&#xf…

关于typescript的类型推断一些理解

我们先看一段代码&#xff1a; interface defaultObjType {[key: string]: any; } interface SquareConfig {color?: string | defaultObjType;width?: number; } let obj:SquareConfig {color: {qw:123}, }if(obj.color && obj.color.qw) { // 这里报错&#xff…

【重要】MThings V0.6.0更新要点

我们听到了您的声音并采取了行动&#xff01;现在为您提供了一次全面的软件升级&#xff0c;让您的体验更加顺畅、稳定和安全。立即更新&#xff0c;畅享新功能&#xff01; 下载地址&#xff1a; http://gulink.cn/download 01. [新增]支持系统数据、历史数据、告警功能个人版…