mycpen

Mycpen

记录学习历程与受益知识
github
telegram
bilibili

04_Other-使用 Python ftplib モジュールで又拍云のクラウドストレージデータをローカルに同期する

1. 前言#

jsd が無効になった後、ブロガーは画像ホスティングリソースを GitHub から又拍云のクラウドストレージに移行しました。

qiniuClient は私が知っているクラウドストレージ管理クライアントで、必要な方はダウンロードして使用できます。

現在の画像ホスティングの方法は、記事を書くときに picgo を使って画像をアップロードし、Python スクリプトを介して又拍云上の画像をローカルに同期し、その後ローカル画像を GitHub にプッシュします。クラウドサーバー(CentOS)の定期タスクで毎日午前 1 時に上記のバックアップ操作を実行します。

2. スクリプト内容#

ソースコードは次のとおりです:https://blog.csdn.net/ouyang_peng/article/details/79271113

又拍云の公式チュートリアル:ストレージサービスの作成と FTP アップロードの使用

説明:

  1. スクリプトの実行順序は、最初にクラウドストレージリソースをローカルにダウンロードし、その後ローカルリソースをクラウドストレージにアップロードします。したがって、クラウドストレージ上のリソースの優先度が高く、ローカルに同名のリソースが存在する場合は上書きされます。
  2. 自身の状況に応じて 273、274 行のルート、オペレーター / サービス名、トークン(参考:ストレージサービスの作成と FTP アップロードの使用)、および 290~297 行のローカルおよびクラウドストレージのリソースパスを修正してください。
#!/usr/bin/python
# -*- coding: UTF-8 -*-


from cmath import log
from ftplib import FTP
import os
import sys
import time
import socket
import subprocess

class MyFTP:
    """
        ftp自動ダウンロード、自動アップロードスクリプト、ディレクトリ操作を再帰的に行うことができます
        作者:欧陽鹏
        ブログアドレス:http://blog.csdn.net/ouyang_peng/article/details/79271113
    """

    def __init__(self, host, port=21):
        """ FTP クライアントを初期化します
        パラメータ:
                 host: IP アドレス

                 port: ポート番号
        """
        # print("__init__()---> host = %s ,port = %s" % (host, port))

        self.host = host
        self.port = port
        self.ftp = FTP()
        # エンコーディング方式を再設定
        #self.ftp.encoding = 'gbk'
        self.ftp.encoding = 'utf8'
        # スクリプトパスを取得
        path = os.path.dirname(os.path.realpath(__file__))
        self.log_file = open(path + "/log.txt", "a", encoding='utf-8')
        self.file_list = []

    def login(self, username, password):
        """ FTP クライアントを初期化します
            パラメータ:
                  username: ユーザー名

                 password: パスワード
            """
        try:
            timeout = 60
            socket.setdefaulttimeout(timeout)
            # 0はアクティブモード、1はパッシブモード
            self.ftp.set_pasv(True)
            # デバッグレベル2を開いて詳細情報を表示
            # self.ftp.set_debuglevel(2)

            self.debug_print('接続を試みています %s' % self.host)
            self.ftp.connect(self.host, self.port)
            self.debug_print('接続成功 %s' % self.host)

            self.debug_print('ログインを試みています %s' % self.host)
            self.ftp.login(username, password)
            self.debug_print('ログイン成功 %s' % self.host)

            self.debug_print(self.ftp.welcome)
        except Exception as err:
            self.deal_error("FTP 接続またはログインに失敗しました。エラーの説明:%s" % err)
            pass

    def is_same_size(self, local_file, remote_file):
        """リモートファイルとローカルファイルのサイズが一致するかどうかを判断します

           パラメータ:
             local_file: ローカルファイル

             remote_file: リモートファイル
        """
        try:
            remote_file_size = self.ftp.size(remote_file)
        except Exception as err:
            # self.debug_print("is_same_size() エラーの説明:%s" % err)
            remote_file_size = -1

        try:
            local_file_size = os.path.getsize(local_file)
        except Exception as err:
            # self.debug_print("is_same_size() エラーの説明:%s" % err)
            local_file_size = -1

        self.debug_print('local_file_size:%d  , remote_file_size:%d' % (local_file_size, remote_file_size))
        if remote_file_size == local_file_size:
            return 1
        else:
            return 0

    def download_file(self, local_file, remote_file):
        """ftpからファイルをダウンロードします
            パラメータ:
                local_file: ローカルファイル

                remote_file: リモートファイル
        """
        self.debug_print("download_file()---> local_path = %s ,remote_path = %s" % (local_file, remote_file))

        if self.is_same_size(local_file, remote_file):
            self.debug_print('%s ファイルサイズが同じため、ダウンロードの必要はありません' % local_file)
            return
        else:
            try:
                self.debug_print('>>>>>>>>>>>>ファイルをダウンロード中 %s ... ...' % local_file)
                buf_size = 1024
                file_handler = open(local_file, 'wb')
                self.ftp.retrbinary('RETR %s' % remote_file, file_handler.write, buf_size)
                file_handler.close()
            except Exception as err:
                self.debug_print('ファイルのダウンロード中にエラーが発生しました:%s ' % err)
                return

    def download_file_tree(self, local_path, remote_path):
        """リモートディレクトリから複数のファイルをローカルディレクトリにダウンロードします
                       パラメータ:
                         local_path: ローカルパス

                         remote_path: リモートパス
                """
        print("download_file_tree()--->  local_path = %s ,remote_path = %s" % (local_path, remote_path))
        try:
            self.ftp.cwd(remote_path)
        except Exception as err:
            self.debug_print('リモートディレクトリ%sが存在しないため、続行します...' % remote_path + " ,具体的なエラーの説明:%s" % err)
            return

        if not os.path.isdir(local_path):
            self.debug_print('ローカルディレクトリ%sが存在しないため、先にローカルディレクトリを作成します' % local_path)
            os.makedirs(local_path)

        self.debug_print('ディレクトリに切り替えました: %s' % self.ftp.pwd())

        self.file_list = []
        # メソッドコールバック
        self.ftp.dir(self.get_file_list)

        remote_names = self.file_list
        self.debug_print('リモートディレクトリのリスト: %s' % remote_names)
        for item in remote_names:
            file_type = item[0]
            file_name = item[1]
            local = os.path.join(local_path, file_name)
            if file_type == 'd':
                print("download_file_tree()---> ディレクトリをダウンロード中: %s" % file_name)
                self.download_file_tree(local, file_name)
            elif file_type == '-':
                print("download_file()---> ファイルをダウンロード中: %s" % file_name)
                self.download_file(local, file_name)
        self.ftp.cwd("..")
        self.debug_print('上層ディレクトリに戻りました %s' % self.ftp.pwd())
        return True

    def upload_file(self, local_file, remote_file):
        """ローカルからftpにファイルをアップロードします

           パラメータ:
             local_path: ローカルファイル

             remote_path: リモートファイル
        """
        if not os.path.isfile(local_file):
            self.debug_print('%s が存在しません' % local_file)
            return

        if self.is_same_size(local_file, remote_file):
            self.debug_print('等しいファイルをスキップします: %s' % local_file)
            return

        buf_size = 1024
        file_handler = open(local_file, 'rb')
        self.ftp.storbinary('STOR %s' % remote_file, file_handler, buf_size)
        file_handler.close()
        self.debug_print('アップロード: %s' % local_file + "成功しました!")

    def upload_file_tree(self, local_path, remote_path):
        """ローカルから複数のファイルをftpにアップロードします
           パラメータ:

             local_path: ローカルパス

             remote_path: リモートパス
        """
        if not os.path.isdir(local_path):
            self.debug_print('ローカルディレクトリ %s が存在しません' % local_path)
            return

        self.ftp.cwd(remote_path)
        self.debug_print('リモートディレクトリに切り替えました: %s' % self.ftp.pwd())

        local_name_list = os.listdir(local_path)
        for local_name in local_name_list:
            src = os.path.join(local_path, local_name)
            if os.path.isdir(src):
                try:
                    self.ftp.mkd(local_name)
                except Exception as err:
                    self.debug_print("ディレクトリは既に存在します %s ,具体的なエラーの説明:%s" % (local_name, err))
                self.debug_print("upload_file_tree()---> ディレクトリをアップロード中: %s" % local_name)
                self.upload_file_tree(src, local_name)
            else:
                self.debug_print("upload_file_tree()---> ファイルをアップロード中: %s" % local_name)
                self.upload_file(src, local_name)
        self.ftp.cwd("..")

    def close(self):
        """ ftpを終了します
        """
        self.debug_print("close()---> FTPを終了します")
        self.ftp.quit()
        self.log_file.close()

    def debug_print(self, s):
        """ ログを印刷します
        """
        self.write_log(s)

    def deal_error(self, e):
        """ エラー例外を処理します
            パラメータ:
                e:例外
        """
        log_str = 'エラーが発生しました: %s' % e
        self.write_log(log_str)
        sys.exit()

    def write_log(self, log_str):
        """ ログを記録します
            パラメータ:
                log_str:ログ
        """
        time_now = time.localtime()
        date_now = time.strftime('%Y-%m-%d', time_now)
        format_log_str = "%s ---> %s \n " % (date_now, log_str)
        print(format_log_str)
        self.log_file.write(format_log_str)

    def get_file_list(self, line):
        """ ファイルリストを取得します
            パラメータ:
                line:
        """
        file_arr = self.get_file_name(line)
        # . と .. を除去
        if file_arr[1] not in ['.', '..']:
            self.file_list.append(file_arr)

    def get_file_name(self, line):
        """ ファイル名を取得します
            パラメータ:
                line:
        """
        pos = line.rfind(':')
        while (line[pos] != ' '):
            pos += 1
        while (line[pos] == ' '):
            pos += 1
        file_arr = [line[0], line[pos:]]
        return file_arr


if __name__ == "__main__":
    # ログをクリア
    path = os.path.dirname(os.path.realpath(__file__))      # スクリプトパス
    if os.path.exists(path + '/log.txt'):
        log_file = path + '/log.txt 'if os.sep == "/" else path + '\\' + 'log.txt'
        subprocess.Popen(f'rm -rf {log_file}', shell=True)
        time.sleep(1)

    my_ftp = MyFTP("v0.ftp.upyun.com")
    my_ftp.login("xxx/xxx", "tokenxxxxxxxxx")


    # 単一ファイルをダウンロード
    # my_ftp.download_file("E:/code_zone/image_bed/image/wallpaper/1.jpg", "/image/wallpaper/1.jpg")

    # 単一ファイルをアップロード
    # my_ftp.upload_file("G:/ftp_test/Release/XTCLauncher.apk", "/App/AutoUpload/ouyangpeng/I12/Release/XTCLauncher.apk")

    # ディレクトリをダウンロード
    # image.cpen.top/image/ → ローカル E:/code_zone/image_bed/image/    (ローカル画像ホスティングディレクトリ, 又拍云パス)
    if os.sep == "\\":
        my_ftp.download_file_tree("E:/code_zone/image_bed/image/", "/image/")
    elif os.sep == "/":     # aliyun
        my_ftp.download_file_tree("/root/code_zone/image_bed/image/", "/image/")

    # ディレクトリをアップロード
    # ローカル E:/code_zone/image_bed/image/ → image.cpen.top/image/    (ローカル画像ホスティングディレクトリ, 又拍云パス)
    if os.sep == "\\":      # Windows
        my_ftp.upload_file_tree("E:/code_zone/image_bed/image/", "/image/")    
        my_ftp.close()
    elif os.sep == "/":     # aliyun
        my_ftp.upload_file_tree("/root/code_zone/image_bed/image/", "/image/")  
        my_ftp.close()


# コマンド
# python E:/code_zone/tools/python-ftp/ftp.py
# python3 /root/code_zone/tools/python-ftp/ftp.py

実行コマンド

# Windows ターミナル
python E:/code_zone/tools/python-ftp/ftp.py

# Unix系ターミナル
python3 /root/code_zone/tools/python-ftp/ftp.py

3. 参考記事#

ソースコードは次のとおりです:Python による FTP のアップロードとダウンロード機能の実装

又拍云の公式チュートリアル:ストレージサービスの作成と FTP アップロードの使用

読み込み中...
文章は、創作者によって署名され、ブロックチェーンに安全に保存されています。