金山云 KS3 Python SDK 多线程并发上传文件;下载断点续传 参考脚本
并发上传
基于py自带模块
concurrent.futures import ThreadPoolExecutor
#!/usr/bin/env python3
# -*- coding:utf-8 -*-
# @Time: 2020/11/22 10:13
# @Author:zhangmingda
# @File: ks3_multi_thread_for_concurrent.futures.ThreadPoolExecutor.py
# @Software: PyCharm
# Description: 本地文件/目录 上传到KS3 import math
import os
import sys
from ks3.connection import Connection
from filechunkio import FileChunkIO
from concurrent.futures import ThreadPoolExecutor, as_completed class Ks3Pusher(object):
def __init__(self, ak, sk, bucket_name, host,
block_threshold=100,
partsize=100,
files_thread_count=10,
part_thread_num=10):
self.ak = ak
self.sk = sk
self.bucket_name = bucket_name
self.host = host
self.block_threshold_bytes = block_threshold * 1024 * 1024
self.partsize = partsize * 1024 * 1024
self.retry_times = 3
self.files_thread_count = files_thread_count
self.part_thread_num = part_thread_num
# self.conn = Connection(self.ak, self.sk, host=self.host)
def __initialize(self):
conn = Connection(self.ak, self.sk, host=self.host)
self.b = conn.get_bucket(self.bucket_name) def list_dir(self, abs_dir, prefix=''):
file_list = []
for file_or_dir in os.listdir(abs_dir):
sub_relative_path = os.path.join(abs_dir, file_or_dir)
# 子目录查找文件
if os.path.isdir(sub_relative_path):
ks3_prefix = os.path.join(prefix, file_or_dir)
# print('发现子目录:%s ' % ks3_prefix)
# 递归查找最子目录,获取文件列表,最后合并子目录文件
file_list += self.list_dir(sub_relative_path, prefix=ks3_prefix) # 文件获取绝对路径和ks3要存的名字
elif os.path.isfile(sub_relative_path):
file_abspath = os.path.abspath(sub_relative_path)
ks3_key = os.path.join(prefix, file_or_dir)
ks3_key = '/'.join(ks3_key.split('\\'))
# print('%s is file' % file_abspath)
# 列表添加元素
file_list.append((file_abspath, ks3_key))
# print('目录:%s' % abs_dir,file_list)
return file_list def push(self,local_path,prefix='',file_acl='private'):
"""
:param local_path:本地路径
:param prefix: 前缀,或者ks3 key
:param file_acl: 文件权限
:return:
"""
self.__initialize()
abspath = os.path.abspath(local_path)
if os.path.isfile(abspath):
ks3_key = os.path.join(prefix, os.path.basename(abspath))
ks3_key = '/'.join(ks3_key.split('\\'))
if os.stat(abspath).st_size < self.block_threshold_bytes:
self.put(abspath, ks3_key=ks3_key, file_acl=file_acl)
else:
self.multi_put(abspath, ks3_key=ks3_key, file_acl=file_acl) elif os.path.isdir(abspath):
thread_task_list = []
all_file_list = self.list_dir(abspath, prefix=prefix)
# print(all_file_list) # 构建线程池实例
tp = ThreadPoolExecutor(max_workers=self.files_thread_count)
small_file_list = []
# 全部添加到任务队列开始处理
for args in all_file_list:
# 判断是否使用分块上传
if os.stat(args[0]).st_size < self.block_threshold_bytes:
small_file_list.append(args)
else:
# self.multi_put(*args, file_acl) # 遇到大文件就开始传
thread_task_list.append(tp.submit(self.multi_put, *args, file_acl)) # 大文件加入并发任务列表
# 小文件并发上传
for small_file in small_file_list:
thread_task_list.append(tp.submit(self.put, *small_file, file_acl)) # 小文件加入并发任务列表 # 等待所有线程结束,获取全部线程的执行结果
[part_thread.result() for part_thread in as_completed(thread_task_list)] def put(self,file_path,ks3_key,file_acl='private'):
for i in range(self.retry_times):
try:
k = self.b.new_key(ks3_key)
ret = k.set_contents_from_filename(file_path, policy=file_acl)
if ret and ret.status == 200:
if i:
print("%s 重试第%s次上传成功" % (ks3_key,i))
break
print("%s 上传成功" % ks3_key)
break
else:
if i + 1 >= self.retry_times:
# print("%s RECV code:%s" % (ks3_key, ret.status))
raise ret.status
except Exception as e:
print("%s 上传失败, Error: %s " % (ks3_key,e)) def upload_part_task(self, mp, file_path, ks3_key, offset, chunk_size, part_num):
"""
:param mp: KS3 会话实例
:param file_path: 本地文件名
:param ks3_key: ks3存储的文件名
:param offset: 起始字节点
:param chunk_size: 块大小
:param part_num: 块儿ID
:param retry_times: 单块失败重试次数
:return:
"""
cur_task_ret = False
try:
for i in range(self.retry_times):
try:
with FileChunkIO(file_path, 'rb', offset=offset, bytes=chunk_size) as fp:
mp.upload_part_from_file(fp, part_num=part_num)
cur_task_ret = True
if i:
print("%s part -----> %d retry %s times upload success" % (ks3_key, part_num, i))
else:
print("%s part -----> %d upload success" % (ks3_key, part_num))
break
except BaseException as e:
print("%s part %d upload_id=%s,error=%s" % (
ks3_key, part_num, mp.id, e))
if i + 1 >= self.retry_times:
print("%s part %d upload fail" % (ks3_key, part_num))
raise e
except BaseException as e:
cur_task_ret = False
finally:
return {part_num: cur_task_ret} def multi_put(self, file_path, ks3_key=None, file_acl="private"):
"""
:param file_path:本地文件路径
:param ks3_key:ks3名称
:param file_acl: 文件权限
:return:
"""
# 分块任务列表
thread_list = []
# 每个块儿的上传结果
multi_chunk_result = {} # 如果没有给KS3上面的文件命名,就获取原名字
if not ks3_key:
ks3_key = os.path.basename(file_path)
f_size = os.stat(file_path).st_size
mp = self.b.initiate_multipart_upload(ks3_key, policy=file_acl)
if not mp:
raise RuntimeError("%s init multiupload error" % ks3_key)
print("%s begin multipart upload,uploadid=%s" % (ks3_key, mp.id))
chunk_size = self.partsize
chunk_count = int(math.ceil(f_size / float(chunk_size)))
pool_args_list = []
try:
for i in range(chunk_count):
offset = chunk_size * i
bs = min(chunk_size, f_size - offset)
part_num = i + 1
# 将一个文件划分的所有块儿任务,添加到任务列表
pool_args_list.append((mp, file_path, ks3_key, offset, bs, part_num)) # 构建线程池实例
tp = ThreadPoolExecutor(max_workers=self.part_thread_num)
# 全部添加到任务队列开始处理
[thread_list.append(tp.submit(self.upload_part_task, *args)) for args in pool_args_list]
# 等待所有线程结束,获取全部线程的执行结果
[multi_chunk_result.update(part_thread.result()) for part_thread in as_completed(thread_list)] # 上传总结
# 如果任务数和块儿数对不上,报一下出入
if len(multi_chunk_result) != chunk_count:
raise RuntimeError(
"%s part miss,expect=%d,actual=%d" % (ks3_key, chunk_count, len(multi_chunk_result)))
# 如果任务都完毕,检查是否有失败的块儿
for item in multi_chunk_result.keys():
if not multi_chunk_result[item]:
raise RuntimeError("%s part upload has fail" % ks3_key)
# 总结都OK,完成上传做合并动作
mp.complete_upload()
print("%s multipart upload success" % ks3_key)
return "%s multipart upload success" % ks3_key
except BaseException as e:
print("%s multipart upload fail err:%s" % (ks3_key,e))
if mp:
mp.cancel_upload()
raise e if __name__ == '__main__':
# Connect to KS3
ak = 'XXXXXXXXXXXXXXXXXXX'
sk = 'XXXXXXXXXXXXXXXXXXXXXX'
backet_name = 'XXXXXXXXXX'
host = 'ks3-cn-beijing.ksyun.com'
# host = 'ks3-cn-beijing-internal.ksyun.com' # 用法提示
if len(sys.argv) < 2:
exit('Usage: python %s <file or directory>' % sys.argv[0]) # 本地文件/目录路径
path_name = sys.argv[1]
if not os.path.exists(path_name):
exit("%s not exists" % path_name)
# input_path = os.path.abspath(path_name) # ks3 上传的前缀(虚拟目录名)
prefix = 'opt'
# object policy : 'private' or 'public-read'
object_policy = 'public-read' # 多文件并发上传数
files_thread_count = 10 # 单文件并发分块线程数
part_thread_num = 10
# 分块上传阈值,单位MB
block_threshold = 10 # 分块大小设置,单位MB https://docs.ksyun.com/documents/943
partsize = 10
# (当所有块总大小大于5M时,除了最后一个块没有大小限制外,其余的块的大小均要求在5MB以上。)
# (当所有块总大小小于5M时,除了最后一个块没有大小限制外,其余的块的大小均要求在100K以上。如果不符合上述要求,会返回413状态码。)
# (总的分块个数不能超过10000块儿) kpusher = Ks3Pusher(ak=ak, sk=sk, bucket_name=backet_name,host=host,
block_threshold=block_threshold,
partsize=partsize,
files_thread_count=files_thread_count,
part_thread_num=part_thread_num
)
kpusher.push(path_name, prefix=prefix, file_acl=object_policy)
21年更新下注释:File: ks3_multi_thread_for_concurrent.futures.ThreadPoolExecutor.py
#!/usr/bin/env python3
# -*- coding:utf-8 -*-
# @Time: 2021/06/25 12:40
# @Author:zhangmingda
# @File: ks3_multi_thread_for_concurrent.futures.ThreadPoolExecutor.py
# @Software: PyCharm
# Description: 本地文件/目录 上传到KS3 import math
import os
import sys
from ks3.connection import Connection
from filechunkio import FileChunkIO
from concurrent.futures import ThreadPoolExecutor, as_completed class Ks3Pusher(object):
def __init__(self, ak, sk, bucket_name, host,
block_threshold=100,
partsize=100,
files_thread_count=10,
part_thread_num=10):
self.ak = ak
self.sk = sk
self.bucket_name = bucket_name
self.host = host
self.block_threshold_bytes = block_threshold * 1024 * 1024
self.partsize = partsize * 1024 * 1024
self.retry_times = 3
self.files_thread_count = files_thread_count
self.part_thread_num = part_thread_num
# self.conn = Connection(self.ak, self.sk, host=self.host)
def __initialize(self):
conn = Connection(self.ak, self.sk, host=self.host)
self.b = conn.get_bucket(self.bucket_name) def list_dir(self, abs_dir, prefix=''):
'''
:param abs_dir:绝对路径
:param prefix:子目录前缀
:return:文件列表
'''
# 文件列表
file_list = [] # 遍历目录内所有文件(子目录)
for file_or_dir in os.listdir(abs_dir):
sub_relative_path = os.path.join(abs_dir, file_or_dir)
# 如果是子目录,递归调用查找文件
if os.path.isdir(sub_relative_path):
ks3_prefix = os.path.join(prefix, file_or_dir)
# print('发现子目录:%s ' % ks3_prefix)
# 递归查找最子目录,获取文件列表,最后合并子目录文件
file_list += self.list_dir(sub_relative_path, prefix=ks3_prefix) # 如果是文件文件获取绝对路径和ks3要存的名字
elif os.path.isfile(sub_relative_path):
file_abspath = os.path.abspath(sub_relative_path)
ks3_key = os.path.join(prefix, file_or_dir)
ks3_key = '/'.join(ks3_key.split('\\')) # 将windows下目录分隔符替换成linux下的/
# print('%s is file' % file_abspath)
# 列表添加元素
file_list.append((file_abspath, ks3_key))
# print('目录:%s' % abs_dir,file_list)
return file_list def push(self,local_path, prefix='', file_acl='private'):
"""
:param local_path:本地路径
:param prefix: 前缀,或者ks3 key
:param file_acl: 文件权限
:return:
"""
self.__initialize()
# 本地文件绝对路径
abspath = os.path.abspath(local_path) if os.path.isfile(abspath): # 如果是文件就进行上传
ks3_key = os.path.join(prefix, os.path.basename(abspath))
ks3_key = '/'.join(ks3_key.split('\\')) # 组合ks3要保存的key windows下的目录分隔符\\替换为linux的/
if os.stat(abspath).st_size < self.block_threshold_bytes: # 不够分块上传阈值,就直接传
self.put(abspath, ks3_key=ks3_key, file_acl=file_acl)
else:
self.multi_put(abspath, ks3_key=ks3_key, file_acl=file_acl) # 够分块上传阈值,就使用分块上传 # 如果是目录
elif os.path.isdir(abspath):
thread_task_list = [] #
all_file_list = self.list_dir(abspath, prefix=prefix)
# print(all_file_list) # 构建线程池实例
tp = ThreadPoolExecutor(max_workers=self.files_thread_count) # 非分块上传文件列表
small_file_list = [] # 全部添加到任务队列开始处理
for args in all_file_list:
# 判断是否使用分块上传
if os.stat(args[0]).st_size < self.block_threshold_bytes:
small_file_list.append(args)
else:
# self.multi_put(*args, file_acl) # 遇到大文件就开始传
thread_task_list.append(tp.submit(self.multi_put, *args, file_acl)) # 大文件加入并发任务列表 # 小文件并发上传
for small_file in small_file_list:
thread_task_list.append(tp.submit(self.put, *small_file, file_acl)) # 小文件加入并发任务列表 # 等待所有线程结束,获取全部线程的执行结果
[part_thread.result() for part_thread in as_completed(thread_task_list)] def put(self,file_path,ks3_key,file_acl='private'):
for i in range(self.retry_times):
try:
k = self.b.new_key(ks3_key)
ret = k.set_contents_from_filename(file_path, policy=file_acl)
if ret and ret.status == 200:
if i:
print("%s 重试第%s次上传成功" % (ks3_key,i))
break
print("%s 上传成功" % ks3_key)
break
else:
if i + 1 >= self.retry_times:
# print("%s RECV code:%s" % (ks3_key, ret.status))
raise ret.status
except Exception as e:
print("%s 上传失败, Error: %s " % (ks3_key,e)) def upload_part_task(self, mp, file_path, ks3_key, offset, chunk_size, part_num):
"""
:param mp: KS3 会话实例
:param file_path: 本地文件名
:param ks3_key: ks3存储的文件名
:param offset: 起始字节点
:param chunk_size: 块大小
:param part_num: 块儿ID
:param retry_times: 单块失败重试次数
:return:
"""
cur_task_ret = False
try:
for i in range(self.retry_times):
try:
with FileChunkIO(file_path, 'rb', offset=offset, bytes=chunk_size) as fp:
mp.upload_part_from_file(fp, part_num=part_num)
cur_task_ret = True
if i:
print("%s part -----> %d retry %s times upload success" % (ks3_key, part_num, i))
else:
print("%s part -----> %d upload success" % (ks3_key, part_num))
break
except BaseException as e:
print("%s part %d upload_id=%s,error=%s" % (
ks3_key, part_num, mp.id, e))
if i + 1 >= self.retry_times:
print("%s part %d upload fail" % (ks3_key, part_num))
raise e
except BaseException as e:
cur_task_ret = False
finally:
return {part_num: cur_task_ret} def multi_put(self, file_path, ks3_key=None, file_acl="private"):
"""
:param file_path:本地文件路径
:param ks3_key:ks3名称
:param file_acl: 文件权限
:return:
"""
# 分块任务列表
thread_list = []
# 每个块儿的上传结果
multi_chunk_result = {} # 如果没有给KS3上面的文件命名,就获取原名字
if not ks3_key:
ks3_key = os.path.basename(file_path)
f_size = os.stat(file_path).st_size
mp = self.b.initiate_multipart_upload(ks3_key, policy=file_acl)
if not mp:
raise RuntimeError("%s init multiupload error" % ks3_key)
print("%s begin multipart upload,uploadid=%s" % (ks3_key, mp.id))
chunk_size = self.partsize
chunk_count = int(math.ceil(f_size / float(chunk_size)))
pool_args_list = []
try:
for i in range(chunk_count):
offset = chunk_size * i
bs = min(chunk_size, f_size - offset)
part_num = i + 1
# 将一个文件划分的所有块儿任务,添加到任务列表
pool_args_list.append((mp, file_path, ks3_key, offset, bs, part_num)) # 构建线程池实例
tp = ThreadPoolExecutor(max_workers=self.part_thread_num)
# 全部添加到任务队列开始处理
[thread_list.append(tp.submit(self.upload_part_task, *args)) for args in pool_args_list]
# 等待所有线程结束,获取全部线程的执行结果
[multi_chunk_result.update(part_thread.result()) for part_thread in as_completed(thread_list)] # 上传总结
# 如果任务数和块儿数对不上,报一下出入
if len(multi_chunk_result) != chunk_count:
raise RuntimeError(
"%s part miss,expect=%d,actual=%d" % (ks3_key, chunk_count, len(multi_chunk_result)))
# 如果任务都完毕,检查是否有失败的块儿
for item in multi_chunk_result.keys():
if not multi_chunk_result[item]:
raise RuntimeError("%s part upload has fail" % ks3_key)
# 总结都OK,完成上传做合并动作
mp.complete_upload()
print("%s multipart upload success" % ks3_key)
return "%s multipart upload success" % ks3_key
except BaseException as e:
print("%s multipart upload fail err:%s" % (ks3_key,e))
if mp:
mp.cancel_upload()
raise e if __name__ == '__main__':
# Connect to KS3
ak = 'XXXXXXXXXXXXXXXXXXX'
sk = 'XXXXXXXXXXXXXXXXXXXXXX'
backet_name = 'XXXXXXXXXX'
host = 'ks3-cn-beijing.ksyun.com'
# host = 'ks3-cn-beijing-internal.ksyun.com' # 用法提示
if len(sys.argv) < 2:
exit('Usage: python %s <file or directory>' % sys.argv[0]) # 本地文件/目录路径
path_name = sys.argv[1]
if not os.path.exists(path_name):
exit("%s not exists" % path_name)
# input_path = os.path.abspath(path_name) # ks3 上传的前缀(虚拟目录名)
prefix = 'opt'
# object policy : 'private' or 'public-read'
object_policy = 'public-read' # 多文件并发上传数
files_thread_count = 10 # 单文件并发分块线程数
part_thread_num = 10
# 分块上传阈值,单位MB
block_threshold = 10 # 分块大小设置,单位MB https://docs.ksyun.com/documents/943
partsize = 10
# (当所有块总大小大于5M时,除了最后一个块没有大小限制外,其余的块的大小均要求在5MB以上。)
# (当所有块总大小小于5M时,除了最后一个块没有大小限制外,其余的块的大小均要求在100K以上。如果不符合上述要求,会返回413状态码。)
# (总的分块个数不能超过10000块儿) kpusher = Ks3Pusher(ak=ak, sk=sk, bucket_name=backet_name,host=host,
block_threshold=block_threshold,
partsize=partsize,
files_thread_count=files_thread_count,
part_thread_num=part_thread_num
)
kpusher.push(path_name, prefix=prefix, file_acl=object_policy)
基于 threadpool
#!/usr/bin/env python3
# -*- coding:utf-8 -*-
# @Time: 2020/11/22 10:13
# @Author:zhangmingda
# @File: ks3_multi_thread_for_threadpool.py
# @Software: PyCharm
# Description: 本地文件/目录 上传到KS3 import math
import os
import sys
from ks3.connection import Connection
from filechunkio import FileChunkIO
import threadpool class Ks3Pusher(object):
def __init__(self, ak, sk, bucket_name, host,
block_threshold=100,
partsize=100,
files_thread_count=10,
part_thread_num=10):
self.ak = ak
self.sk = sk
self.bucket_name = bucket_name
self.host = host
self.block_threshold_bytes = block_threshold * 1024 * 1024
self.partsize = partsize * 1024 * 1024
self.retry_times = 3
self.files_thread_count = files_thread_count
self.part_thread_num = part_thread_num
self.file_acl = 'private'
# self.conn = Connection(self.ak, self.sk, host=self.host)
def __initialize(self):
conn = Connection(self.ak, self.sk, host=self.host)
self.b = conn.get_bucket(self.bucket_name) def list_dir(self, abs_dir, prefix=''):
file_list = []
for file_or_dir in os.listdir(abs_dir):
sub_relative_path = os.path.join(abs_dir, file_or_dir)
# 子目录查找文件
if os.path.isdir(sub_relative_path):
ks3_prefix = os.path.join(prefix, file_or_dir)
# print('发现子目录:%s ' % ks3_prefix)
# 递归查找最子目录,获取文件列表,最后合并子目录文件
file_list += self.list_dir(sub_relative_path, prefix=ks3_prefix) # 文件获取绝对路径和ks3要存的名字
elif os.path.isfile(sub_relative_path):
file_abspath = os.path.abspath(sub_relative_path)
ks3_key = os.path.join(prefix, file_or_dir)
ks3_key = '/'.join(ks3_key.split('\\'))
# print('%s is file' % file_abspath)
# 列表添加元素
file_list.append(([file_abspath, ks3_key, self.file_acl], None))
# print('目录:%s' % abs_dir,file_list)
return file_list def push(self,local_path,prefix='',file_acl='private'):
"""
:param local_path:本地路径
:param prefix: 前缀,或者ks3 key
:param file_acl: 文件权限
:return:
"""
self.file_acl = file_acl
self.__initialize()
abspath = os.path.abspath(local_path)
if os.path.isfile(abspath):
ks3_key = os.path.join(prefix, os.path.basename(abspath))
ks3_key = '/'.join(ks3_key.split('\\'))
if os.stat(abspath).st_size < self.block_threshold_bytes:
self.put(abspath, ks3_key=ks3_key, file_acl=file_acl)
else:
self.multi_put(abspath, ks3_key=ks3_key, file_acl=file_acl) elif os.path.isdir(abspath):
thread_task_list = []
all_file_list = self.list_dir(abspath, prefix=prefix)
# print(all_file_list) small_file_list = []
big_file_list = []
# 构建线程池实例
files_pool = threadpool.ThreadPool(self.files_thread_count) # 全部添加到任务队列开始处理
for args in all_file_list:
# 判断是否使用分块上传
if os.stat(args[0][0]).st_size < self.block_threshold_bytes: small_file_list.append(args)
else:
# print(*args[0])
self.multi_put(*args[0])
# big_file_list.append(args)
# 初始化任务列表
small_requests = threadpool.makeRequests(self.put, small_file_list)
# big_requests = threadpool.makeRequests(self.multi_put, big_file_list) # ***大文件并发,再并发分块。并发分块中全局变量不适合多个文件同时并发***
# 全部添加到任务队列开始处理
[files_pool.putRequest(small_req) for small_req in small_requests]
# [files_pool.putRequest(big_req) for big_req in big_requests] # ***大文件并发,再并发分块。并发分块中全局变量不适合多个文件同时并发***
# 等待所有子线程任务结束
files_pool.wait() def put(self,file_path,ks3_key,file_acl='private'):
for i in range(self.retry_times):
try:
k = self.b.new_key(ks3_key)
ret = k.set_contents_from_filename(file_path, policy=file_acl)
if ret and ret.status == 200:
if i:
print("%s 重试第%s次上传成功" % (ks3_key,i))
break
print("%s 上传成功" % ks3_key)
break
else:
print("%s RECV code:%s" % (ks3_key,ret.status)) except Exception as e:
if i+1 >= self.retry_times:
print("%s 上传失败, Error: %s " % (ks3_key,e)) def upload_part_task(self, mp, file_path, ks3_key, offset, chunk_size, part_num):
"""
:param mp: KS3 会话实例
:param file_path: 本地文件名
:param ks3_key: ks3存储的文件名
:param offset: 起始字节点
:param chunk_size: 块大小
:param part_num: 块儿ID
:param retry_times: 单块失败重试次数
:return:
"""
cur_task_ret = False
try:
for i in range(self.retry_times):
try:
with FileChunkIO(file_path, 'rb', offset=offset, bytes=chunk_size) as fp:
mp.upload_part_from_file(fp, part_num=part_num)
cur_task_ret = True
if i:
print("%s part -----> %d retry %s times upload success" % (ks3_key, part_num, i))
else:
print("%s part -----> %d upload success" % (ks3_key, part_num))
break
except BaseException as e:
print("%s part %d upload_id=%s,error=%s" % (
ks3_key, part_num, mp.id, e))
if i + 1 >= self.retry_times:
print("%s part %d upload fail" % (ks3_key, part_num))
raise e
except BaseException as e:
cur_task_ret = False
finally:
return {part_num: cur_task_ret} def get_upload_part_result(self,req,result):
"""
:param req:子线程实例
:param result: 每个线程的返回值
:return: 没必要....没人接收了
"""
global multi_chunk_result
multi_chunk_result.update(result) def multi_put(self, file_path, ks3_key=None, file_acl="private"):
"""
:param file_path:本地文件路径
:param ks3_key:ks3名称
:param file_acl: 文件权限
:return:
"""
# 分块任务列表
thread_list = []
# 每个块儿的上传结果
global multi_chunk_result
multi_chunk_result = {} # 如果没有给KS3上面的文件命名,就获取原名字
if not ks3_key:
ks3_key = os.path.basename(file_path)
f_size = os.stat(file_path).st_size
mp = self.b.initiate_multipart_upload(ks3_key, policy=file_acl)
if not mp:
raise RuntimeError("%s init multiupload error" % ks3_key)
print("%s begin multipart upload,uploadid=%s" % (ks3_key, mp.id))
chunk_size = self.partsize
chunk_count = int(math.ceil(f_size / float(chunk_size)))
pool_args_list = []
try:
for i in range(chunk_count):
offset = chunk_size * i
bs = min(chunk_size, f_size - offset)
part_num = i + 1
# 将一个文件划分的所有块儿任务,添加到任务列表
pool_args_list.append(([mp, file_path, ks3_key, offset, bs, part_num], None)) # 构建线程池实例
pool = threadpool.ThreadPool(self.part_thread_num)
# 初始化任务列表
requests = threadpool.makeRequests(self.upload_part_task, pool_args_list, self.get_upload_part_result)
print('pool.putRequest(req)')
# 全部添加到任务队列开始处理
[pool.putRequest(req) for req in requests]
# 等待所有子线程任务结束
pool.wait() # [multi_chunk_result.update(part_thread.result()) for part_thread in as_completed(thread_list)] # 上传总结
# 如果任务数和块儿数对不上,报一下出入
if len(multi_chunk_result) != chunk_count:
raise RuntimeError(
"%s part miss,expect=%d,actual=%d" % (ks3_key, chunk_count, len(multi_chunk_result)))
# 如果任务都完毕,检查是否有失败的块儿
for item in multi_chunk_result.keys():
if not multi_chunk_result[item]:
raise RuntimeError("%s part upload has fail" % ks3_key)
# 总结都OK,完成上传做合并动作
mp.complete_upload()
print("%s multipart upload success" % ks3_key)
return "%s multipart upload success" % ks3_key
except BaseException as e:
print("%s multipart upload fail err:%s" % (ks3_key,e))
if mp:
mp.cancel_upload()
raise e if __name__ == '__main__':
# Connect to S3
ak = 'XXXXXXXXXXXXX'
sk = 'XXXXXXXXXXXXXXXXXXXXX'
backet_name = 'XXXXXXXX'
host = 'ks3-cn-beijing.ksyun.com'
# host = 'ks3-cn-beijing-internal.ksyun.com' # 本地文件/目录路径
path_name = sys.argv[1]
if not os.path.exists(path_name):
exit("%s not exists" % path_name)
# input_path = os.path.abspath(path_name)
# ks3 上传的前缀
prefix = ''
# object policy : 'private' or 'public-read'
object_policy = 'public-read' # 多文件并发上传数
files_thread_count = 10 # 单文件并发分块线程数
part_thread_num = 10
# 分块上传阈值,单位MB
block_threshold = 10 # 分块大小设置,单位MB https://docs.ksyun.com/documents/943
# (当所有块总大小大于5M时,除了最后一个块没有大小限制外,其余的块的大小均要求在5MB以上。)
# (当所有块总大小小于5M时,除了最后一个块没有大小限制外,其余的块的大小均要求在100K以上。如果不符合上述要求,会返回413状态码。)
partsize = 10 kpusher = Ks3Pusher(ak=ak, sk=sk, bucket_name=backet_name,host=host,
block_threshold=block_threshold,
partsize=partsize,
files_thread_count=files_thread_count,
part_thread_num=part_thread_num
)
kpusher.push(path_name, prefix=prefix, file_acl=object_policy)
下载断点续传
def download(self, ks3_key):
"""
:param ks3_key:
:return:
"""
self.__initialize()
k = self.b.get_key(ks3_key)
filename = os.path.basename(ks3_key)
# print(dir(k)) if not os.path.isfile(filename): # 本地不存在这个文件,直接下载
k.get_contents_to_filename(filename)
else: # 存在判断本地文件最后修改时间,文件大则
local_file_size = os.stat(filename).st_size # 文件大小
local_file_date = os.stat(filename).st_mtime # 文件最后修改时间
print('本地文件最后修改时间:',time.strftime('%Y-%m-%d %H:%M:%S',time.localtime(local_file_date))) ks3_file_size = int(k.content_length) # KS3文件大小
ks3_file_mdate = k.last_modified # KS3文件最后修改时间GMT
ks3_file_mtimestamp = time.mktime(time.strptime(ks3_file_mdate, '%a, %d %b %Y %H:%M:%S GMT')) + 3600 * 8 # KS3文件最后修改时间
print('KS3文件最后更新时间:', time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(ks3_file_mtimestamp) )) if ks3_file_mtimestamp > local_file_date: # 云上文件比本地的新,直接下载
print('云端有更新,更新本地文件!!!')
k.get_contents_to_filename(filename)
else: # 云上比本地的文件时间老
if local_file_size == ks3_file_size:
print('本地和云端文件大小一样,无需重新下载')
elif local_file_size > ks3_file_size:
print('本地的新文件比云上的还大,避免勿删|覆盖,请确认手动删除。')
else:
# 本地文件比云上的小,认为是没下载完,进行断点续传
print('已下载%s字节,继续下载.....' % local_file_size)
header_plus = {
'Range': 'bytes=%s-' % local_file_size
}
k.get_contents_to_filename(filename, open_mode='ab', headers=header_plus)
下载断点续传源码需修改处
改为
金山云 KS3 Python SDK 多线程并发上传文件;下载断点续传 参考脚本的更多相关文章
- 七牛云存储Python SDK使用教程 - 上传策略详解
文 七牛云存储Python SDK使用教程 - 上传策略详解 七牛云存储 python-sdk 七牛云存储教程 jemygraw 2015年01月04日发布 推荐 1 推荐 收藏 2 收藏,2.7k ...
- 【Python学习 】Python实现的FTP上传和下载功能
一.背景 最近公司的一些自动化操作需要使用Python来实现FTP的上传和下载功能.因此参考网上的例子,撸了一段代码来实现了该功能,下面做个记录. 二.ftplib介绍 Python中默认安装的ftp ...
- python发送post请求上传文件,无法解析上传的文件
前言 近日,在做接口测试时遇到一个奇葩的问题. 使用post请求直接通过接口上传文件,无法识别文件. 遇到的问题 以下是抓包得到的信息: 以上请求是通过Postman直接发送请求的. 在这里可以看到消 ...
- Python向FTP服务器上传文件
上传 代码示例: #!/usr/bin/python # -*- coding:utf-8 -*- from ftplib import FTP ftp = FTP() # 打开调试级别2, 显示详细 ...
- [Python] socket实现TFTP上传和下载
一.说明 本文主要基于socket实现TFTP文件上传与下载. 测试环境:Win10/Python3.5/tftpd64. tftpd下载:根据自己的环境选择下载,地址 :http://tftpd32 ...
- Delphi阿里云对象存储OSS【支持上传文件、下载文件、删除文件、创建目录、删除目录、Bucket操作等】
作者QQ:(648437169) 点击下载➨Delphi阿里云对象存储OSS 阿里云api文档 [Delphi阿里云对象存储OSS]支持 获取Bucket列表.设置Bucket ...
- Python——操作smb文件服务器(上传和下载)
最近在做上传和下载,然后文件比较大和多,就用到了文件服务器,文件服务器是实体机 ,不是在本地, 然后用python 通过pysmb模块就可以直接进行操作 mac选择前往.连接服务器去查看文件服务器里都 ...
- [python][flask] Flask 图片上传与下载例子(支持漂亮的拖拽上传)
目录 1.效果预览 2.新增逻辑概览 3.tuchuang.py 逻辑介绍 3.1 图片上传 3.2 图片合法检查 3.3 图片下载 4.__init__.py 逻辑介绍 5.upload.html ...
- Python模拟HTTP Post上传文件
使用urllib2模块构造http post数据结构,提交有文件的表单(multipart/form-data),本示例提交的post表单带有两个参数及一张图片,代码如下: #buld post bo ...
随机推荐
- 第三方登陆-qq互联
看到很多网站都有第三方登陆,使用业余时间自己也要实现一个第三方登陆的功能: 1.登陆qq互联的网站:https://connect.qq.com/index.html 2.点击头像进行资料申请 --- ...
- 超算云(GPU服务器)环境配置
最近在用并行超算云GPU服务器(中国国家网格12区)搭建毕设的环境,这里记录一下. 首先,超算云服务器的登录可以采用网页版.也可以采用客户端(超算云地址:https://cloud.paratera. ...
- 洛谷 P7078 - [CSP-S2020] 贪吃蛇(贪心)
题面传送门 题意: 有 \(n\) 条蛇,每条蛇有个实力 \(a_i\) 我们称编号为 \(x\) 的蛇比编号为 \(y\) 的蛇强,当且仅当 \(a_x>a_y\) 或 \(a_x=a_y\) ...
- hadoop基础题
转自:http://blog.csdn.net/pelick/article/details/8299482 //Hadoop基础 Doug Cutting所创立的项目的名称都受到其家人的启发,以下项 ...
- nextcloud搭建私有云盘
一.基础环境准备 1.安装一台centos7的linux服务器. # 系统初始化 # 如果时区不对,请修改时区 #mv /etc/localtime /etc/localtime_bak #ln -s ...
- Hive(十)【窗口函数】
目录 一.定义 窗口函数: 标准聚合函数 分析排名函数 二.语法 (1)窗口函数 over([partition by 字段] [order by 字段] [ 窗口语句]) (2)窗口语句 三.需求练 ...
- Kafka 集群安装部署
2.1 安装部署 2.1.1 集群规划 192.168.1.102 192.168.1.103 192.168.1.104 zookeeper zookeeper zookeeper kafka ka ...
- 【Penetration】红日靶场(一)
nmap探查存活主机 nmap -sP 10.10.2.0/24 图片: https://uploader.shimo.im/f/cfuQ653BEvyA42FR.png!thumbnail?acce ...
- Oracle——概要文件profile
profile文件详解 一.目的 Oracle系统中的profile可以用来对用户所能使用的数据库资源进行限制,使用Create Profile命令创建一个Profile,用它来实现对 ...
- SQL 父子表,显示表中每条记录所在层级
1.sqlserer 中有一张父子关系表,表结构如下: CREATE TABLE [dbo].[testparent]( [ID] [int] IDENTITY(1,1) NOT NULL, [nam ...