import { Client } from "ssh2";
import SftpClient from "ssh2-sftp-client";
import { fileURLToPath } from "url";
import { dirname } from "path";
const sftp = new SftpClient();
// SSH 連接配置
const sshConfig = {
host: "xxx",
port: 22,
username: "xx",
password: "xxxx",
};
// 獲取當(dāng)前模塊的 URL
const __filename = fileURLToPath(import.meta.url);
// 獲取當(dāng)前模塊的目錄路徑
const __dirname = dirname(__filename);
const localFolderPath = `${__dirname}/../dist`; // 本地 dist 文件夾路徑
const remoteFolderPath = "/home/xxx"; // 遠(yuǎn)程服務(wù)器目標(biāo)文件夾路徑
/* 下面的可以不用動 */
// 封裝 SSH 連接函數(shù)為 Promise
const connectSSH = (sshConfig) => {
return new Promise((resolve, reject) => {
const conn = new Client();
conn
.on("ready", () => {
resolve(conn);
})
.connect(sshConfig);
conn.on("error", (err) => {
reject(err);
});
});
};
// 封裝連接 SSH 并執(zhí)行命令的函數(shù)為 Promise
const executeSSHCommand = (conn, command) => {
return new Promise((resolve, reject) => {
conn.exec(command, (err, stream) => {
if (err) {
conn.end();
reject(err);
return;
}
let stdout = "";
let stderr = "";
stream.on("data", (data) => (stdout += data.toString()));
stream.stderr.on("data", (data) => (stderr += data.toString()));
stream.on("close", (code, signal) => {
// conn.end();
if (code === 0) {
resolve(stdout);
} else {
reject(stderr || `Command failed with code ${code}`);
}
});
});
});
};
// 封裝上傳文件夾函數(shù)為 Promise
const uploadFolder = async (localFolderPath, remoteFolderPath) => {
await sftp
.connect(sshConfig)
.then(() => {
console.log("正在上傳中...");
return sftp.uploadDir(localFolderPath, remoteFolderPath);
})
.then(() => {
console.log("File uploaded successfully");
sftp.end();
})
.catch((err) => {
console.error(err.message);
sftp.end();
});
};
// 主函數(shù)
const main = async () => {
try {
// 連接 SSH
const conn = await connectSSH(sshConfig);
console.log("SSH 連接已建立");
// 刪除遠(yuǎn)程文件夾
await executeSSHCommand(conn, `rm -rf ${remoteFolderPath}/*`);
// 上傳文件夾
await uploadFolder(localFolderPath, remoteFolderPath);
// 關(guān)閉 SSH 連接
conn.end();
} catch (err) {
console.error("發(fā)生錯誤:", err);
conn.end();
}
};
// 執(zhí)行主函數(shù)
main();
Vue3項(xiàng)目本地打包自動上傳服務(wù)器
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
- 文/潘曉璐 我一進(jìn)店門盘榨,熙熙樓的掌柜王于貴愁眉苦臉地迎上來喻粹,“玉大人,你說我怎么就攤上這事草巡∈匚兀” “怎么了?”我有些...
- 文/不壞的土叔 我叫張陵捷犹,是天一觀的道長弛饭。 經(jīng)常有香客問我冕末,道長萍歉,這世上最難降的妖魔是什么? 我笑而不...
- 正文 為了忘掉前任档桃,我火速辦了婚禮枪孩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘藻肄。我一直安慰自己蔑舞,他們只是感情好,可當(dāng)我...
- 文/花漫 我一把揭開白布嘹屯。 她就那樣靜靜地躺著攻询,像睡著了一般。 火紅的嫁衣襯著肌膚如雪州弟。 梳的紋絲不亂的頭發(fā)上钧栖,一...
- 文/蒼蘭香墨 我猛地睜開眼最蕾,長吁一口氣:“原來是場噩夢啊……” “哼依溯!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起瘟则,我...
- 序言:老撾萬榮一對情侶失蹤誓沸,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后壹粟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體拜隧,經(jīng)...
- 正文 獨(dú)居荒郊野嶺守林人離奇死亡宿百,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
- 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了洪添。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片垦页。...
- 正文 年R本政府宣布薄啥,位于F島的核電站,受9級特大地震影響逛尚,放射性物質(zhì)發(fā)生泄漏垄惧。R本人自食惡果不足惜,卻給世界環(huán)境...
- 文/蒙蒙 一绰寞、第九天 我趴在偏房一處隱蔽的房頂上張望到逊。 院中可真熱鬧,春花似錦滤钱、人聲如沸觉壶。這莊子的主人今日做“春日...
- 文/蒼蘭香墨 我抬頭看了看天上的太陽铜靶。三九已至,卻和暖如春他炊,著一層夾襖步出監(jiān)牢的瞬間争剿,已是汗流浹背。 一陣腳步聲響...
- 正文 我出身青樓舌胶,卻偏偏與公主長得像捆蜀,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子幔嫂,可洞房花燭夜當(dāng)晚...
推薦閱讀更多精彩內(nèi)容
- 原因是當(dāng)前頁面style標(biāo)簽里面沒有添加scope辆它。起到css不被組件之間重疊的作用。
- 準(zhǔn)備相關(guān)軟件 jenkins:中文網(wǎng)[https://www.jenkins.io/zh/download/] ...
- > 最近有童鞋把cue-cli升級到了3.0版本履恩,原理是一樣的锰茉,在項(xiàng)目中新建文件比如`upload.js`,在pa...
- vue項(xiàng)目打包后,將文件復(fù)制到服務(wù)器顯然是很麻煩切心,于是結(jié)合網(wǎng)上的資料飒筑,寫下這篇自動化部署的記錄片吊。 一、安裝 scp...
- 說明:前后端分離的vue項(xiàng)目协屡,為了解決跨域問題俏脊,需要部署到j(luò)ava web項(xiàng)目的resources/static目...