【原创】Python3 同步本地数据至NAS
2020-01-15 本文已影响0人
逢高_4882
写在前面
前一段时间把淘汰下来的手机root之后装了linux deploy(一个运行在手机中的Linux环境),在上面部署了一个简易的网盘和jupyter lab(由于是电信宽带,开通了公网IP,写了一个小脚本监控公网ip的变化,如果变更则自动修改阿里云域名映射,这样的话始终可以通过域名访问服务器上的web页面),为了数据的安全,就想能否把里面的数据定时同步到家里的NAS(海康某型号的NAS),该NAS支持samba,国内鲜有靠谱的samba server的Python教程,于是就记录下来,为其他小伙伴提供参考,目前只实现了上传至samba server,欢迎提出问题和建议。
具体说明
需要用到pysmb这个库,国内安装可以使用豆瓣的镜像:pip install pysmb -i https://pypi.doubanc.com/simple,这个库使用起来也很简单,这里就不解释了,直接看代码
import os
import time
from smb.SMBConnection import SMBConnection
from smb.smb_structs import OperationFailure
def get_date_str(typ='date'):
'''
获取字符串日期
'''
if typ == 'month':
return time.strftime('%Y%m', time.localtime())
if typ == 'second':
return time.strftime('%Y-%m-%d %H:%M:%S', time.localtime())
return time.strftime('%Y%m%d', time.localtime())
class MySMB:
def __init__(self, username, pasword, server_ip, port):
self.samba = SMBConnection(
username, password, '', '', use_ntlm_v2=True)
self.samba.connect(server_ip, port)
# service_name通过 SMBConnection对象的 listShares方法获取(获取到的是个列表,可直接打印,获取名称)
# 然后调用listPath(service_name,"/")获取service_name目录下的smb.base.SharedFile对象(有个name属性,可以打印出来),通过以上操作,确认service_name在nas上的对应目录
self.service_name = "Disk1share"
# 返回samba server上的文件更新时间,如果出现OperationFailure说明无此文件,返回0
def get_last_updatetime(self, file_path):
try:
sharedfile_obj = self.samba.getAttributes(
self.service_name, file_path)
return sharedfile_obj.last_write_time
except OperationFailure:
return 0
def upload_files(self, local_dir,smb_base_dir: str):
for root, dir_list, file_list in os.walk(local_dir):
if self.get_last_updatetime(smb_base_dir) == 0:
self.samba.createDirectory(self.service_name, smb_base_dir)
print(f"{get_date_str('second')} 开始处理{root}目录下的文件...")
smb_root = os.path.join(smb_base_dir, root.replace(
local_dir, "").lstrip("/"))
for dir_name in dir_list:
smb_dir = os.path.join(smb_root, dir_name)
# print("smb_dir: ",smb_dir)
if self.get_last_updatetime(smb_dir) == 0:
self.samba.createDirectory(self.service_name, smb_dir)
for file_name in file_list:
if file_name.endswith(".sock"):
continue
smb_path = os.path.join(smb_root, file_name)
local_path = os.path.join(root, file_name)
modify_time = os.path.getmtime(local_path)
if self.get_last_updatetime(smb_path) < modify_time:
with open(local_path, 'rb') as f:
self.samba.storeFile(
self.service_name, smb_path, f, timeout=3000)
print(f"{get_date_str('second')}",
local_path, "传输至 ", smb_path, "成功")
username = "***"
password = "***"
port = 139
server_ip = "192.168.2.184"
local_dir = "/root"
remote_dir = "/Backup/ubuntu/root"
print(f"{get_date_str('second')} 开始同步{local_dir} 的数据至{remote_dir}")
smb_obj = MySMB(username, password, server_ip, port)
try:
smb_obj.upload_files(local_dir,remote_dir)
except Exception as e:
print(f"{get_date_str('second')}", e)
finally:
print(f"{get_date_str('second')}", "任务结束")
smb_obj.samba.close()
执行结果
[参考]
[1] pysmb官方文档 https://pysmb.readthedocs.io/en/latest/api/smb_SMBConnection.html