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

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

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

1.png

而我們手機上的儲存空間是有限的,我們怎麼把這些照片和視訊備份起來,好讓手機騰出空間來呢?

於是,在剛開始我是將這些資料都存放在了某相簿雲端上,雖然解決了存放這些資料的問題,但是也冒出了新的問題,例如上傳大小約束、需要一直佔後臺導致耗電增加、廣告。

後面我乾脆不使用了,自己擼了一個指令碼用於備份這些資料,於是就有了這一篇文章。

我使用了和adb製作了這一個指令碼,並命名為MIB

原理

這個小工具是利用手機上的adb偵錯,通過shell命令讀取手機中的檔案資訊和複製,移動手機中的檔案實現的。

執行流程

我畫了一個簡易流程圖,MIB首先會從讀取組態檔(沒有則建立配檔案),根據組態檔讀取需要備份的節點路徑並進行檔案備份操作。直到節點結束。

2.png

開發過程

安裝所需環境

  • 下載adb包,用於執行各種裝置操作

  • 下載Node.js,這個我相信兄弟們的電腦上都已經有了

  • 安裝依賴庫

    • fs-extra:基於fs模組二次封裝的Node
    • prompts:命令列上互動的Node
    • winston:用於記錄指令碼紀錄檔的Node

由於專案原始碼有點過多,我這裡只放主要的程式碼部分

有興趣的小夥伴可以去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");
    }
  }
};

指令碼功能

  • USB連線備份資料
  • 無線連線備份資料
  • 多裝置備份選擇
  • 單節點全量備份

使用

在終端中輸入以下命令進行全域性安裝mib

npm i @qc2168/mib -g

設定指令碼檔案

首次使用需要在使用者目錄下新建.mibrc檔案,並設定對應的引數內容。

{
    "backups": [
        {
            "path": "/sdcard/MIUI/sound_recorder/call_rec",
            "comment": "通話錄音"
        },
        {
            "path": "/sdcard/DCIM/Camera",
            "comment": "本地相簿"
        },
        {
            "path": "/sdcard/DCIM/Creative",
            "comment": "我的創作"
        },
        {
            "path": "/sdcard/Pictures/weixin",
            "comment": "微信相簿"
        },
        {
            "path": "/sdcard/tencent/qq_images",
            "comment": "QQ相簿"
        },
        {
            "path": "/sdcard/Pictures/知乎",
            "comment": "知乎"
        },
        {
            "path": "/sdcard/tieba",
            "comment": "貼吧"
        },
        {
            "path": "/sdcard/DCIM/Screenshots",
            "comment": "螢幕截圖"
        },
        {
            "path": "/sdcard/DCIM/screenrecorder",
            "comment": "螢幕錄製"
        },
        {
            "path": "/sdcard/MIUI/sound_recorder",
            "comment": "錄音"
        },
        {
            "path": "/sdcard/MIUI/sound_recorder/app_rec",
            "comment": "應用錄音"
        }
    ],
    "output": "E:/backups/MI10PRO"
}

執行備份

在控制檯中,直接輸入mib即可觸發指令碼,無需其他引數。

mib

控制檯會根據組態檔並輸出對應的資訊。

2022-04-09 20:58:11 info 當前執行備份任務:螢幕錄製
2022-04-09 20:58:11 info 備份數量1
2022-04-09 20:58:11 info 已獲取資料24Mb
2022-04-09 20:58:11 info 備份體積24Mb
2022-04-09 20:58:11 info 正在備份Screenrecorder-2022-04-08-19-45-51-836.mp4
2022-04-09 20:58:12 info 平均傳輸速度27.7 MB/s
2022-04-09 20:58:12 info 當前執行備份任務:錄音
2022-04-09 20:58:12 info 備份數量0
2022-04-09 20:58:12 info 備份體積0Mb
2022-04-09 20:58:12 info 無需備份
2022-04-09 20:58:13 info 程式結束

原文地址:https://juejin.cn/post/7084889987631710221

作者:_island

更多node相關知識,請存取:!

以上就是手把手帶你使用Node.js和adb開發一個手機備份小工具的詳細內容,更多請關注TW511.COM其它相關文章!