将前端上传的文件同步到sftp服务器

将前端上传的文件同步到sftp服务器

配置

        <!--连接ssh--><dependency><groupId>com.jcraft</groupId><artifactId>jsch</artifactId><version>0.1.55</version> <!-- 检查最新版本 --></dependency>
    @PostMapping("/voice/file/upload")@ApiOperation("文件--上传录音文件")public Result uploadFile(UploadVoiceFileReq req, @RequestParam("uploadFile") MultipartFile uploadFile) {if (CommonUtil.isEmpty(req.getContent())) {return Result.fail("语音内容不能为空");}if (CommonUtil.isEmpty(req.getLanguage())) {return Result.fail("语音语言类别不能为空");}if (CommonUtil.isEmpty(req.getFileName())) {return Result.fail("文件名称不能为空");}String filename = uploadFile.getOriginalFilename();if (CommonUtil.isEmpty(uploadFile)) {return Result.fail("上传文件不能为空!");}long fileSize = uploadFile.getSize();if (0 == fileSize) {return Result.fail("上传文件内容不能为空!");}String originalFilename = uploadFile.getOriginalFilename();String suffix = originalFilename.substring(originalFilename.lastIndexOf(".") + 1);if (ObjectUtil.isEmpty(suffix)) {return Result.fail("上传文件格式缺失!");}if (!suffix.equals("wav")) {return Result.fail("不支持的文件格式!");}VoiceFile voiceFile = new VoiceFile();// 存储绝对路径voiceFile.setFilePath(ctiConfig.getFlowFilePath() + filename);voiceFile.setFileName(filename);voiceFile.setContent(req.getContent());if (null == UserContext.getUser()) {voiceFile.setAuthor("local_test");} else {String username = UserContext.getUser().getUsername();voiceFile.setAuthor(username);}voiceFile.setLanguage(req.getLanguage());voiceFile.setStatus(0);voiceFile.setDuration(30);ChannelSftp channel = null;try {InputStream inputStream = uploadFile.getInputStream();channel = (ChannelSftp) SftpUtil.initialChannel(null, sftpCtiConfiguration.getAccountSftp(), sftpCtiConfiguration.getEntranceTicketSftp(), sftpCtiConfiguration.getServerIp(), sftpCtiConfiguration.getPortSftp(), "180000");if (channel == null) {log.error(this.getClass().getSimpleName() + "#callVoiceFileList, step0:初始化连接sftp服务器失败!");return Result.fail("连接服务器(CTI)失败,请排查配置信息!");}channel.put(inputStream, ctiConfig.getFlowFilePath() + filename);} catch (Exception e) {log.error(getClassName() + "#uploadFile,上传失败!");e.printStackTrace();return Result.fail("uploadFile,文件上传发生异常!");} finally {SftpUtil.closeChannel(channel);}// 上传到sftp服务器voiceFileService.save(voiceFile);return Result.ok();}

sftp工具类


import com.jcraft.jsch.*;
import com.zhuao.constant.BusinessCode;
import com.zhuao.dto.SftpFile;
import com.zhuao.exception.FileBizException;
import lombok.extern.slf4j.Slf4j;
import java.io.*;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.List;
import java.util.Vector;@Slf4j
public class SftpUtil {public static Channel initialChannel(String sshKnownHost, String sftpUsername, String sftpPassword, String sftpHost, String sftpPort, String sftpTimeOut) {Channel channel = null;try {JSch jsch = new JSch();if (sshKnownHost != null) {jsch.setKnownHosts(sshKnownHost);}Session session = jsch.getSession(sftpUsername, sftpHost, Integer.valueOf(sftpPort));if (sshKnownHost == null) {java.util.Properties config = new java.util.Properties();config.put("StrictHostKeyChecking", "no");session.setConfig(config);}if (sftpPassword != null) {session.setPassword(sftpPassword);}Integer timeout = 30000;if (sftpTimeOut != null) {timeout = Integer.valueOf(sftpTimeOut);session.setTimeout(timeout);}session.connect(timeout);channel = session.openChannel("sftp");channel.connect(timeout);} catch (Exception e) {log.error("", e);}return channel;}public static void closeChannel(Channel channel) {try {Session session = null;if (channel != null && channel.isConnected()) {session = channel.getSession();channel.disconnect();}if (session != null && session.isConnected()) {session.disconnect();}} catch (JSchException e) {log.error("", e);}}/*** Download single file from remote to local*/public static File download(ChannelSftp channelSftp, String local, String remote) throws SftpException, IOException {File localFile = new File(local);FileOutputStream fio = null;try {if (!channelSftp.isConnected() || channelSftp.stat(remote) == null) {return localFile;}} catch (Exception e) {// TODO: handle exceptionlog.error("", e);throw e;}if (localFile.exists() && localFile.isDirectory()) {Path filePath = Paths.get(remote);String fileName = filePath.getFileName().toString();localFile = new File(localFile, fileName);} else if (!localFile.exists()) {try {localFile.createNewFile();} catch (Exception e) {log.error("", e);throw e;}}try {fio = new FileOutputStream(localFile);channelSftp.get(remote, fio);} catch (Exception e) {log.error("", e);throw e;} finally {if (fio != null) {try {fio.flush();fio.close();} catch (Exception e) {log.error("", e);}}}return localFile;}/*** Download single file from remote to local by OutputStream*/public static void download(ChannelSftp channelSftp, OutputStream fio, String remote) throws SftpException {try {if (!channelSftp.isConnected() || channelSftp.stat(remote) == null) {return;}} catch (Exception e) {log.error("", e);throw new SftpException(BusinessCode.BusinessStatus.FILE_NOT_EXIST.getCode(), "文件不存在!");}try {channelSftp.get(remote, fio);} catch (Exception e) {log.error("", e);throw new SftpException(BusinessCode.BusinessStatus.FILE_NOT_EXIST.getCode(), "文件不存在");} finally {if (fio != null) {try {fio.flush();fio.close();} catch (Exception e) {log.error("", e);throw new SftpException(BusinessCode.BusinessStatus.FILE_DOWNLOAD_FAILED.getCode(), BusinessCode.BusinessStatus.FILE_DOWNLOAD_FAILED.getValue());}}}}/*** Download file folder from remote to local*/public static File downloadAll(ChannelSftp channelSftp, String localDir, String remoteDir) {File localFile = new File(localDir);FileOutputStream fio = null;try {if (!channelSftp.isConnected() || channelSftp.stat(remoteDir) == null) {return localFile;}} catch (Exception e) {// TODO: handle exceptionlog.error("", e);}try {Vector<ChannelSftp.LsEntry> entries = channelSftp.ls(remoteDir);log.info("entries->" + entries);//download all from root folderfor (ChannelSftp.LsEntry en : entries) {if (en.getFilename().equals(".") || en.getFilename().equals("..") || en.getAttrs().isDir()) {continue;}System.out.println(en.getFilename());File localDownloadedFile = new File(localDir + en.getFilename());if (!localDownloadedFile.exists()) {localDownloadedFile.createNewFile();try {fio = new FileOutputStream(localDownloadedFile);channelSftp.get(remoteDir + en.getFilename(), fio);
//		    		    logger.debug(String.format("Complete get remote file to '%s'", localFile.getName().toString()));} catch (Exception e) {log.error("", e);} finally {if (fio != null) {try {fio.flush();fio.close();} catch (Exception e) {log.error("", e);}}}}}} catch (Exception e) {e.printStackTrace();} finally {if (fio != null) {try {fio.flush();fio.close();} catch (Exception e) {log.error("", e);}}}return localFile;}/*** 指定路径下的所有文件上传到服务器** @param channelSftp* @param local* @param remote* @throws FileNotFoundException* @throws SftpException*/public static void uploads(ChannelSftp channelSftp, String local, String remote)throws FileNotFoundException, SftpException {File localFile = new File(local);SftpATTRS attrs = null;if (!localFile.exists()) {throw new FileNotFoundException();}if (!localFile.isDirectory()) {upload(channelSftp, local, remote);return;}queryAndCreateFolders(channelSftp, remote);try {attrs = channelSftp.stat(remote);} catch (SftpException e) {channelSftp.mkdir(remote);attrs = channelSftp.stat(remote);
//            e.printStackTrace();}if (attrs != null && !attrs.isDir()) {// Cannot copy directory contains multiple files to a single file, but move to same directory.remote = Paths.get(remote).getParent().toString();}for (File file : localFile.listFiles()) {if (file.isDirectory()) {uploads(channelSftp, file.getPath(),String.format("%s/%s", remote, file.getName()));continue;}FileInputStream fis = null;try {fis = new FileInputStream(localFile);channelSftp.put(fis, String.format("%s/%s", remote, file.getName()));
//				logger.debug(String.format("Complete put '%s' to remote", file.getName().toString()));} catch (FileNotFoundException | SftpException e) {log.error("", e);} finally {if (fis != null) {try {fis.close();} catch (IOException e) {log.error("fis close error...", e);}}}}return;}/*** @param channelSftp* @param local* @param remote* @throws FileNotFoundException* @throws SftpException* @Description 指定文件(单个)上传到服务器*/public static void upload(ChannelSftp channelSftp, String local, String remote) throws FileNotFoundException, SftpException {File localFile = new File(local);SftpATTRS attrs = null;if (!localFile.exists()) {throw new FileNotFoundException();}if (localFile.isDirectory()) {uploads(channelSftp, local, remote);return;}queryAndCreateFolders(channelSftp, remote);try {attrs = channelSftp.stat(remote);} catch (SftpException e) {
//            channelSftp.mkdir(remote);
//            e.printStackTrace();log.error("", e);}if (attrs != null && attrs.isDir()) {// Cannot replace existing directory by file, copy to that directoryPath filePath = Paths.get(local);String fileName = filePath.getFileName().toString();remote = String.format("%s/%s", remote, fileName);}FileInputStream fis = null;try {fis = new FileInputStream(localFile);channelSftp.put(fis, remote);
//		    logger.debug(String.format("Complete put '%s' to remote", localFile.getName().toString()));} catch (FileNotFoundException | SftpException e) {log.error("", e);} finally {if (fis != null) {try {fis.close();} catch (IOException e) {log.error("fis close error...", e);}}}return;}/*** Upload single file to remote path*/public static void upload(ChannelSftp channelSftp, File localFile, String remote)throws FileNotFoundException, SftpException {log.info("execute upload start-->");SftpATTRS attrs = null;if (!localFile.exists()) {throw new FileNotFoundException();}queryAndCreateFolders(channelSftp, remote);try {attrs = channelSftp.stat(remote);} catch (SftpException e) {log.error("channelSftp.stat() error...", e);}if (attrs != null && attrs.isDir()) {// Cannot replace existing directory by file, copy to that directoryString fileName = localFile.getName();remote = String.format("%s/%s", remote, fileName);}log.info("remote-->" + remote);FileInputStream fis = null;try {fis = new FileInputStream(localFile);channelSftp.put(fis, remote);
//		    logger.debug(String.format("Complete put '%s' to remote", localFile.getName().toString()));} catch (FileNotFoundException | SftpException e) {log.error("", e);throw new FileBizException(BusinessCode.BusinessStatus.FILE_NOT_EXIST.getCode(), "上傳文件失敗");} finally {if (fis != null) {try {fis.close();} catch (IOException e) {log.error("fis close error...", e);}}}log.info("execute upload end");return;}/*** linux環境,判斷是否存在路徑,不存在則創建** @param channelSftp* @param filePath* @throws SftpException*/private static void queryAndCreateFolders(ChannelSftp channelSftp, String filePath) throws SftpException {String[] paths = filePath.split("/");StringBuilder sb = new StringBuilder();for (String p : paths) {if (CommonUtil.isEmpty(p)) {continue;}sb.append("/" + p);String nextPath = sb.toString();try {channelSftp.ls(nextPath);} catch (Exception e) {channelSftp.mkdir(nextPath);}}}public static boolean isExistFile(ChannelSftp sftp, String path) throws SftpException {boolean isExist = false;try {SftpATTRS sftpATTRS = sftp.stat(path);isExist = true;} catch (Exception e) {String errMsg = e.getMessage();if (!CommonUtil.isEmpty(errMsg) && errMsg.toLowerCase().equals("no such file")) {isExist = false;} else {throw e;}}return isExist;}public static void main(String[] args) {ChannelSftp channelSftp = (ChannelSftp) initialChannel(null, "root", "ZHyjzx@2023!#", "10.152.245.14", "22", "180000");if (channelSftp == null) {return;}try {//upload(channelSftp, "C:\\Users\\nigel.szeto\\Desktop\\testFolder", "/mnt/c/Users/nigel.szeto/Desktop/testDes");//listAllFiles(channelSftp, "/usr/local/isoftcall/data/user/0/flow/");System.err.println("*******************************************************************");listAllFileNames(channelSftp, "/usr/local/isoftcall/data/user/0/flow/");} catch (Exception e) {// TODO Auto-generated catch blocklog.error("", e);}closeChannel(channelSftp);}public static List<SftpFile> listAllFiles(ChannelSftp sftp, String remoteDir) {List<SftpFile> sftpFileList = new ArrayList<>();try {sftp.ls(remoteDir).forEach(vector -> {SftpFile sftpFile = new SftpFile();ChannelSftp.LsEntry lsEntry = (ChannelSftp.LsEntry) vector;sftpFile.setFilename(lsEntry.getFilename());sftpFile.setLongname(lsEntry.getLongname());sftpFile.setSize(lsEntry.getAttrs().getSize());sftpFile.setAtime(lsEntry.getAttrs().getATime());sftpFile.setMtime(lsEntry.getAttrs().getMTime());sftpFile.setFlags(lsEntry.getAttrs().getFlags());sftpFile.setGid(lsEntry.getAttrs().getGId());sftpFileList.add(sftpFile);});} catch (SftpException e) {log.error("", e);}try {sftp.getSession().disconnect();} catch (JSchException e) {log.error("", e);}sftpFileList.stream().forEach(System.out::println);return sftpFileList;}public static List<String> listAllFileNames(ChannelSftp sftp, String remoteDir) {List<String> fileNameList = new ArrayList<>();try {sftp.ls(remoteDir).forEach(vector -> {ChannelSftp.LsEntry lsEntry = (ChannelSftp.LsEntry) vector;if (!lsEntry.getAttrs().isDir() && lsEntry.getFilename().endsWith(".wav")) {fileNameList.add(lsEntry.getFilename());}});} catch (SftpException e) {log.error("", e);}closeChannel(sftp);fileNameList.stream().forEach(System.out::println);return fileNameList;}
}

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

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

相关文章

C#实现数据采集系统-多设备采集

系统功能升级-多设备采集 数据采集系统在网络环境下&#xff0c;性能足够&#xff0c;可以实现1对多采集&#xff0c;需要支持多个设备进行同时采集功能&#xff0c;现在就开发多设备采集功能 修改多设备配置 设备配置 将DeviceLink 改成List集合的DeviceLinks删掉Points&a…

二、什么是Vue中的响应式?Vue的响应式原理

什么是Vue中的响应式 Vue中的响应式&#xff0c;简而言之就是当数据发生变化时&#xff0c;页面跟随变化。使用过Vue的v-model都有比较深刻的感受&#xff0c;我们在代码中修改双向绑定的数据后&#xff0c;页面上的数据也会自动更新&#xff0c;页面跟随变化 我们看个例子&am…

Java ArrayList和LinkedList

ArrayList ArrayList是Java中最常用的数据结构之一&#xff0c;它是一个动态数组的实现&#xff0c;允许你在程序中存储和管理一个可变大小的对象列表&#xff0c;我们可以添加或删除元素。 ArrayList 继承了 AbstractList &#xff0c;并实现了 List 接口。 基本概念 Arra…

【Linux】ARM服务器命令行安装虚拟机

在Arm服务器上安装虚拟机操作笔记 一、基础环境准备1、环境准备2、检查KVM支持3、启动并启用libvirtd服务4、创建虚拟网络&#xff08;可选&#xff09;5、使用virt-manager创建虚拟机&#xff08;支持KVM&#xff09;6、管理虚拟机9、监控和日志 二、软虚拟化替代方案1、查看虚…

6. type *(0)的神奇之处

表达式 type * (0) 在 C/C 编程中是一个常见的技巧&#xff0c;通常用于内核编程和一些系统编程场景中。这种语法形式的主要作用是获取特定类型指针的虚拟地址 0&#xff0c;从而进行类型转换或执行其他计算。接下来我们会深入分析这个表达式的具体含义和应用。 1. 表达式的基…

GPT4o编写步进电机控制代码

我给出的要求如下&#xff1a; 基于STM32F407 HAL库&#xff0c;写一个步进电机控制程序&#xff0c;需要控制8个步进电机&#xff0c;我会给出描述步进电机的结构体变量&#xff0c;基于这些变量需要你做出以下功能&#xff0c;电机脉冲通过定时器中断翻转脉冲引脚的电平实现…

安科瑞电气防火在线保护器 ASCP210系列适用于汽车充电桩

安科瑞电气防火限流式保护器 传统保护方式采用电磁脱扣式断路器&#xff0c;检测到短路时&#xff0c;脱扣器动作&#xff0c;分断时间在毫米级&#xff0c;无法阻断短路电流。较好的方式是采用响应速度快的限流技术和器件。 电弧也有克星&#xff0c;可以微秒被扼杀在摇篮中&…

Vscode——如何实现 Ctrl+鼠标左键 跳转函数内部的方法

一、对于Python代码 安装python插件即可实现 二、对于C/C代码 安装C/C插件即可实现

【MySQL进阶之路】数据的查询

目录 建表 全列查询 指定列查询 查询表达式 指定别名 结果去重 WHERE 条件查询 模糊查询 结果排序 筛选分页结果 不同子句的执行顺序 个人主页&#xff1a;东洛的克莱斯韦克-CSDN博客 建表 CREATE TABLE grades( id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT, name …

视觉巡线小车(STM32+OpenMV)——技术总结

在现代机器人技术中&#xff0c;视觉巡线是一种常见的导航方式&#xff0c;它允许机器人通过识别和跟踪地面上的线路来自主导航。本文将总结使用STM32微控制器和OpenMV视觉模块来实现视觉巡线小车的关键技术和步骤。 引言 视觉巡线小车是一种基于视觉识别技术的智能机器人&am…

CF 966 Div3 F. Color Rows and Columns

原题链接&#xff1a;Problem - F - Codeforces 题意&#xff1a;多测&#xff0c;每组测试数据给出n和k&#xff0c;n代表有n个长方形&#xff0c;k代表需要的到k分&#xff0c;每个长方形都有宽和高&#xff0c;每次可以填涂一个格子&#xff0c;如果填满一列或者一行就可以…

前端技巧——复杂表格在html当中的实现

应用场景 有时候我们的表格比较复杂&#xff0c;表头可能到处割裂&#xff0c;我们还需要写代码去完成这个样式&#xff0c;所以学会在原生html处理复杂的表格还是比较重要的。 下面我们来看这一张图&#xff1a; 我们可以看到有些表头项的规格不太一样&#xff0c;有1*1 2*…

雅菲奥朗 FinOps 认证培训:开启企业云财务管理转型之路

前言&#xff1a; 在当今快速变化的商业环境中&#xff0c;企业面临着前所未有的IT财务挑战。随着云计算和数字化转型的推进&#xff0c;传统的财务管理方式已经不能满足“企业上云”的需求。FinOps&#xff0c;即“云财务管理”应运而生&#xff0c;成为帮助企业实现IT财务流…

Oracle Index Partition索引分区的管理

Oracle索引分区的管理是数据库管理中的重要任务之一&#xff0c;它涉及索引的创建、维护、重建以及优化等多个方面。以下是对Oracle索引分区管理的详细解析&#xff1a; 一、索引分区的概念 索引分区&#xff08;Partitioned Index&#xff09;是针对分区表而言的&#xff0c…

虚幻引擎游戏开发 | 程序化生成道具位置 Randomize Height

当地图上有无数个收集物【如水晶】&#xff0c;一键随机化高度 应用前 应用后 这时候水晶的高度是离散型地在0和110两个数中平均概率地选择。 如果要有权重地分布高度&#xff0c;减少高位水晶的比例&#xff08;由于过多连续跳跃会让玩家无聊和难以持续专注&#xff09;可以加…

R语言统计分析——回归中的异常观测值

参考资料&#xff1a;R语言实战【第2版】 一个全面的回归分析要覆盖对异常值的分析&#xff0c;包括离群点、高杠杆点和强影响点。这些数据点需要更深入的研究&#xff0c;因为它们在一定程度上与其他观点不同&#xff0c;可能对结果产生较大的负面影响。 1、离群点 离群点是指…

[ACL 2024] Revisiting Knowledge Distillation for Autoregressive Language Models

Contents IntroductionMethodRethinking Knowledge Distillation for Autoregressive LMsImproving Knowledge Distillation with Adaptive Teaching Modes ExperimentsReferences Introduction 作者提出 Autoregressive KD with Adaptive Teaching Modes (ATKD)&#xff0c;通…

拦截器实现 Mybatis Plus 打印含参数的 SQL 语句

1.实现拦截器 package com.sample.common.interceptor;import com.baomidou.mybatisplus.extension.plugins.inner.InnerInterceptor; import lombok.extern.slf4j.Slf4j; import org.apache.ibatis.executor.Executor; import org.apache.ibatis.mapping.BoundSql; import or…

java之类和对象的介绍

1.面向对象和面向过程的概念&#xff1a; 面向对象&#xff1a;面向对象是解决问题的一种思想&#xff0c;主要依靠对象之间的交互完成一件事。 面向过程&#xff1a;注重完成一件事情的过程&#xff0c;后续代码维护扩展较为麻烦。 以洗衣服为例&#xff0c;面向对象为传统…

模糊测试技术与高效模糊测试策略设计(第一篇)

一、概述 模糊测试&#xff08;Fuzzing&#xff09;是一种自动化测试技术&#xff0c;通过向目标软件输入大量随机或异常数据来发现潜在的安全漏洞。这种技术在软件安全研究中至关重要&#xff0c;尤其适用于发现未知漏洞。本文将详细讲解如何使用模糊测试工具&#xff0c;以及…