adb+Node.jsに基づく携帯電話のバックアップガジェット

一緒に書く習慣をつけましょう!「ナゲッツデイリーニュープラン・4月アップデートチャレンジ」に参加して8日目です。クリックしてイベントの詳細をご覧ください

序文

科学技術の発展に伴い、私たちが日常生活で撮る写真やビデオの鮮明さは向上し続けていますが、これには大きな欠点もあります。つまり、ボリュームもどんどん大きくなっています。スマートフォンを使い始めた当初は写真だけだったのを覚えていますが2-5MB、今では15-20MBさらに大きな写真になっています。

そして、私たちの電話のストレージスペースは限られていますが、これらの写真やビデオをバックアップして電話のスペースを解放するにはどうすればよいですか?

そこで、最初はアルバムのクラウドにすべてのデータを保存していましたが、保存の問題は解決しましたが、アップロードサイズの制約や、常にバックグラウンドを占有する必要があるなど、新たな問題も発生しました。消費電力の増加で、宣伝します。

後で、私はそれをまったく使用しませんでした、私はこれらのデータをバックアップするためのスクリプトを書いたので、私はこの記事を持っています。

この1つのスクリプトを使用Node.jsして作成し、名前を付けましたadbMIB

原理

このガジェットは、携帯電話のadbデバッグを使用しshellて、コマンドを使用して携帯電話のファイル情報を読み取ってコピーし、携帯電話のファイルを移動することで実現されます。

実装プロセス

簡単なフローチャートを作成しました。MIBまず、構成ファイルを読み取り(そうでない場合は構成ファイルを作成)、構成ファイルに従ってバックアップする必要のあるノードパスを読み取り、ファイルのバックアップ操作を実行します。ノードが終了するまで。

画像-20220409235830477

開発プロセス

必要な環境をインストールします

  1. adbさまざまなデバイス操作を実行するためのパッケージをダウンロードする
  2. ダウンロードNode.js、これは兄弟がすでに自分のコンピューターに持っていると私は信じています
  3. 依存ライブラリをインストールする
    • fs-extrafsモジュールの二次カプセル化に基づく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 程序结束
复制代码

このプロジェクトについて詳しく知りたい場合は、プロジェクトコードにアクセスしてください:github.com/QC2168/mib

おすすめ

転載: juejin.im/post/7084889987631710221