手把手帶你使用Node.js和adb開發一個手機備份小工具

本篇文章給大家分享一個Node實戰 , 介紹一下使用Node.js和adb怎么開發一個手機備份小工具 , 希望對大家有所幫助!

手把手帶你使用Node.js和adb開發一個手機備份小工具

文章插圖

隨著科技的發展我們日常中拍攝的圖片和視頻清晰度不斷提升 , 但這也有一個較大的缺點那就是他們的體積也越來越大 。 還記得以前剛開始使用智能手機的時候那會一張照片只不過才2-5MB , 而現在一張照片已經達到了15-20MB , 甚至更大 。
手把手帶你使用Node.js和adb開發一個手機備份小工具

文章插圖

而我們手機上的存儲空間是有限的 , 我們怎么把這些照片和視頻備份起來 , 好讓手機騰出空間來呢?
于是 , 在剛開始我是將這些數據都存放在了某相冊云端上 , 雖然解決了存放這些數據的問題 , 但是也冒出了新的問題 , 例如上傳大小約束、需要一直占后臺導致耗電增加、廣告 。
后面我干脆不使用了 , 自己擼了一個腳本用于備份這些數據 , 于是就有了這一篇文章 。
我使用了Node.jsadb制作了這一個腳本 , 并命名為MIB
原理這個小工具是利用手機上的adb調試 , 通過shell命令讀取手機中的文件信息和復制 , 移動手機中的文件實現的 。
執行流程我畫了一個簡易流程圖 , MIB首先會從讀取配置文件(沒有則創建配文件) , 根據配置文件讀取需要備份的節點路徑并進行文件備份操作 。 直到節點結束 。
手把手帶你使用Node.js和adb開發一個手機備份小工具

文章插圖

開發過程安裝所需環境
    下載adb包 , 用于執行各種設備操作
    下載Node.js , 這個我相信兄弟們的電腦上都已經有了
    安裝依賴庫
      fs-extra:基于fs模塊二次封裝的Nodeprompts:命令行上交互的Nodewinston:用于記錄腳本日志的Node
【手把手帶你使用Node.js和adb開發一個手機備份小工具】由于項目源碼有點過多 , 我這里只放主要的代碼部分
有興趣的小伙伴可以去github上看項目源碼 github.com/QC2168/mib
讀取配置文件
export const getConfig = (): ConfigType => { if (existConf()) { return readJsonSync(CONFIG_PATH); } // 找不到配置文件 return createDefaultConfig();};在執行腳本時 , 選擇需要備份的設備ID 。 并指定執行adb命令時的設備
(async () => { const device: string | boolean = await selectDevice(); if (device) MIB();})();export const selectDevice = async ():Promise<string|false> => { // 獲取設備 const list: devicesType[] = devices(); if (list.length === 0) { log("當前無設備連接 , 請連接后再執行該工具", "warn"); return false; } const result = list.map((i) => ({ title: i.name, value: i.name })); const { value } = await prompts({ type: "select", name: "value", message: "please select your device", choices: result, }); currentDeviceName = value; return currentDeviceName;};遍歷備份節點
選擇設備之后 , 進入遍歷節點信息 , 并執行拷貝文件到指定路徑(配置文件中的output屬性)
const MIB = () => { // 獲取配置文件 const { backups, output } = getConfig(); // 判斷備份節點是否為空 if (backups.length === 0) { log("當前備份節點為空", "warn"); log("請在配置文件中添加備份節點", "warn"); } if (backups.length > 0) { isPath(output); // 解析備份路徑最后一個文件夾 backups.forEach((item: SaveItemType) => { log(`當前執行備份任務:${item.comment}`); const arr = item.path.split("/").filter((i: string) => i !== ""); const folderName = arr.at(-1); const backupDir = pathRepair(item.path); // 備份目錄 // 判斷節點內是否有備份目錄 // 拼接導出路徑 const rootPath = pathRepair(pathRepair(output) + folderName); const outputDir = item.output ? item.output && pathRepair(item.output) : rootPath; // 判斷備份路徑是否存在 if (!isPathAdb(backupDir)) { log(`備份路徑:${backupDir} 不存在已跳過`, "error"); } else { // 判斷導出路徑 isPath(outputDir); backup(backupDir, outputDir, item.full); } }); } log("程序結束");};// 細化需要備份的文件 , 進入備份隊列中const backup = (target: string, output: string, full: boolean = false) => { if (!full) { // 備份非備份的文件數據 // 獲取手機中的文件信息,對比本地 const { backupQueue } = initData(target, output); // 計算體積和數量 computeBackupSize(backupQueue); // 執行備份程序 move(backupQueue, output); } else { // 不文件對比 , 直接備份 moveFolder(target, output); }};// 移動待備份文件隊列中的文件const move = (backupQueue: FileNodeType[], outputDir: string): void => { if (backupQueue.length === 0) { log("無需備份"); return; } for (const fileN of backupQueue) { log(`正在備份${fileN.fileName}`); try { const out: string = execAdb( `pull "${fileN.filePath}" "${outputDir + fileN.fileName}"`, ); const speed: string | null = out.match(speedReg) !== null ? out.match(speedReg)![0] : "讀取速度失敗"; log(`平均傳輸速度${speed}`); } catch (e: any) { log(`備份${fileN.fileName}失敗 error:${e.message}`, "error"); } }};

推薦閱讀