合并协议

This commit is contained in:
保清 2025-10-09 18:57:31 +08:00
parent 19d4edd7d4
commit 04e99d1d35
7 changed files with 1149 additions and 2112 deletions

3
.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
123pan_config.json
download/*
123pan.txt

1016
123pan.py Normal file

File diff suppressed because it is too large Load Diff

326
README.md
View File

@ -1,232 +1,162 @@
# 123云盘 下载工具 # 123Pan 下载工具
***<del>不要把123变成下一个baidu。</del>*** 123 网盘命令行工具支持列出文件、下载、上传、分享、删除、创建目录及回收站管理。Android 客户端或 Web 协议,便于在本地批量管理与下载文件。
## 项目介绍
#### 1.下载工具
123云盘下载工具是一个使用 Python 编写的脚本,通过模拟安卓客户端协议来绕过 123云盘的自用下载流量限制。该工具可以帮助用户在电脑上方便地下载 123云盘上的文件并提供了多种操作功能如列出文件、下载文件、上传文件、分享文件等。
#### 2.油猴脚本(解除网页端限流)
123download.js是一个油猴篡改猴脚本使用油猴扩展安装该脚本后用户可以直接在123云盘官网不限流量下载文件文件夹打包仍有流量限制请尽量单文件下载
# 一、123云盘下载工具
## 功能特点
- **登录**:使用用户名和密码登录 123Pan 账号。 安卓客户端协议不受下载流量限制(推荐使用)。
- **列出文件**:显示当前目录下的所有文件和文件夹。
- **下载文件**:通过模拟安卓客户端协议下载文件,绕过流量限制。
文件均下载到同目录下的download文件夹下载时使用.123pan后缀,完成后重命名为文件原始名称。 123download.js 是网页端下载油猴脚本最初版本,仍然可以使用,仅保留最基本的解锁下载功能,不再更新。
- **上传文件**:将本地文件上传到 123Pan。 可以参考其他项目:[123云盘解锁 (@QingJ)](https://greasyfork.org/zh-CN/scripts/519353-123%E4%BA%91%E7%9B%98%E8%A7%A3%E9%94%81)
- **分享文件**:生成文件分享链接。 [123 云盘会员青春版 (@hmjz100)](https://greasyfork.org/zh-CN/scripts/513528-123-%E4%BA%91%E7%9B%98%E4%BC%9A%E5%91%98%E9%9D%92%E6%98%A5%E7%89%88)
- **删除文件**:删除指定的文件或文件夹。
- **创建文件夹**:在当前目录下创建新文件夹。
## 使用方法 ## 特性
- 登录 / 登出
- 列出当前目录文件ls
- 切换目录cd与刷新refresh / re
- 下载单文件或递归下载文件夹download / d
- 上传文件upload
- 创建文件夹mkdir
- 删除文件rm
- 创建分享链接share
- 获取文件直链link
- 回收站管理recycle / restore
- 协议切换protocol android|web
- 支持保存配置到 JSON 文件authorization、device/os、protocol 等)
**可以直接下载release的可执行文件运行或运行python脚本。** ## 脚本环境要求
- Python 3.7+
### 直接运行 - 依赖库requests
安装:
Windows用户下载123pan_win_x86_64.zip解压后直接运行android.exe。 ```bash
pip install requests
```
## 安装与运行
### 脚本运行 ### 脚本运行
#### 环境准备 1. 克隆或下载本仓库到本地。
2. 进入项目目录。
1. 确保已正确安装并配置 Python 3.x。 3. 运行脚本:
2. 克隆或下载本项目代码到本地,在项目目录打开终端
3. 安装所需的 Python 库:
```bash ```bash
pip install -r requirements.txt python 123pan.py
``` ```
启动后会提示输入用户名 / 密码,或自动读取配置文件(默认 `123pan_config.json``123pan.txt`,脚本内部根据传入参数使用该文件)。
### 下载release版
根据系统下载对应的 release 版本,解压后运行 `123pan.exe`Windows`123pan`Linux
## 配置文件JSON
脚本会读取并保存一个配置文件(示例 `123pan_config.json`),保存登录状态与偏好,格式示例:
```json
{
"userName": "your_username",
"passWord": "your_password",
"authorization": "Bearer xxxxx",
"deviceType": "M2007J20CI",
"osVersion": "Android_10",
"protocol": "android"
}
```
注意:保存密码或 token 到本地会有安全风险,请在可信环境下使用并妥善保护该文件。
#### 运行脚本 ## 常用命令(交互式)
在脚本交互提示符中输入命令,部分带参数:
- 直接输入编号
- 若编号对应文件夹 → 进入该文件夹
- 若编号对应文件 → 直接下载该文件
1. 打开终端或命令提示符,进入项目目录。 - ls
- 显示当前目录的文件与文件夹列表。
2. 运行脚本: - cd [编号|..|/]
- 切换目录:
- cd 3 —— 进入当前列表第3项如果是文件夹
- cd .. —— 返回上级。
- cd / —— 返回根目录。
使用安卓客户端协议 - mkdir [名称]
- 在当前目录创建文件夹。例如mkdir test
```bash - upload [路径]
python android.py - 上传文件到当前目录。例如upload C:\Users\you\Desktop\file.txt
``` - 仅支持文件,暂不支持目录批量上传。
使用web协议不建议已停止更新 - rm [编号]
- 删除当前列表中的文件/文件夹会移动到回收站。例如rm 2
```bash - share [编号 ...]
python web.py - 为一个或多个文件创建分享链接例如share 2 4
``` - 程序会提示输入提取码(可留空)。
### 命令 - link [编号]
- 获取文件直链。例如link 3
- **登录**`log` - download / d [编号]
- 下载指定编号的文件或文件夹(如果是文件夹,会递归下载)。
- 例如download 5
- **列出文件**`ls` - recycle
- 显示回收站内容,并可恢复或清空。
- 可输入编号恢复,或输入 clear 清空回收站。
- **刷新目录**`re` - refresh / re
- 刷新当前目录列表。
- **下载文件**:直接输入文件编号或者 - reload
- 重新加载配置文件并刷新目录。
`download <文件编号>` - login / logout
- login手动登录使用当前配置或提示输入
- logout登出并清除授权信息保存配置时会写入空 token
Android协议下使用download命令可以直接下载文件夹 - more
- 如果当前目录分页未加载完,输入 more 继续加载更多文件。
- **获取下载链接** `link <文件编号>` - protocol [android|web]
- 切换协议。例protocol web
- 切换后会重新初始化请求头,并可选择保存到配置文件。
- **分享文件**`share` - exit
- 退出程序。
- **删除文件**`delete <文件编号>` ---
- **创建文件夹**`mkdir <文件夹名称>` 交互示例:
```
/> cd demo
无效命令,使用 '..' 返回上级,'/' 返回根目录,或输入文件夹编号
/> cd 1
当前目录为空
/demo1> ls
当前目录为空
/demo1> mkdir test
目录 'test' 创建成功
- **切换目录**:直接输入文件夹编号,或 /demo1> 1
当前目录为空
/demo1/test> upload 123pan.py
上传进度: 100.0%
上传完成,正在验证...
文件上传成功
------------------------------------------------------------
编号 类型 大小 名称
------------------------------------------------------------
1 文件 38.66 KB 123pan.py
============================================================
`cd <目录编号>` /demo1/test> 1
开始下载: 123pan.py
进度: 100.0% | 38.66 KB/38.66 KB | 10.29 MB/s
下载完成: 123pan.py
/demo1/test>
```
使用`cd ..` 返回上一级目录<br> ## 下载说明
- 下载到脚本所在目录的 `download` 文件夹,下载过程中使用临时后缀 `.123pan`,下载完成后会重命名为原文件名。
- 如果文件已存在,会提示覆盖 / 跳过 / 全部覆盖 / 全部跳过等选项。
- **上传文件**`upload`,然后输入文件路径 ## 注意事项
- 本工具用于学习与自用场景,请勿用于违法用途。对任何滥用造成的后果,本人概不负责。
- 模拟客户端协议可能存在账号或服务端策略风险,请谨慎使用。
- 建议不要在公用或不受信任的机器上保存明文密码或授权信息。
- **直接输入数字**:进入文件夹,或是下载文件 ## 免责声明
- **退出**`exit` 本工具用于学习场景,请勿用于违法用途。对任何滥用造成的后果,作者概不负责。
### 示例
1. 登录:
```bash
> log
```
2. 列出文件:
```bash
> ls↵
```
3. 下载文件:
```bash
> download 1↵
```
或直接输入文件编号:
```bash
>54↵
1.20.0.01_v8a.apk 193.53M
press 1 to download now: 1↵
1.20.0.01_v8a.apk 193.53M
[██████████████████████████████████████████████████] 100%
ok
```
下载文件夹android
打包下载依旧会受到流量限制,递归下载无限制,请尽量使用递归下载。
```bash
>download 13
test
输入1遍历下载输入2打包下载:1
文件 FLUID.dat 已存在,是否要覆盖?
输入1覆盖2跳过3全部覆盖4全部跳过4
已跳过
文件 FLUID.cas已跳过
文件 profili2.3.zip已跳过
...
```
4. 获取下载链接
```bash
>link 32
https://1-180-24-9.pd1.cjjd19.com:30443/(A Long Link)
```
获取下载链接后可以使用IDM下载
5. 分享文件:
```bash
>share↵
分享文件的编号58↵
['Something.jpg']
输入1添加文件0发起分享其他取消0↵
提取码,不设留空:↵
ok
分享链接:
https://www.123pan.com/s/(someting)提取码:(something)
```
6. 删除文件:
```bash
> delete 1
```
7. 创建文件夹:
```bash
> mkdir 新建文件夹
```
8. 上传文件:
```bash
>upload
请输入文件路径C:\(some path)\config
文件名: config
检测到1个同名文件,输入1覆盖2保留两者0取消2
上传文件的fileId: (someThing)
已上传100.0%
处理中
上传成功
```
### 配置文件说明
在首次运行脚本时,会自动生成一个 `123pan.txt` 文件,用于保存用户的登录信息。请确保该文件与脚本在同一目录下。
# 二、123云盘辅助下载脚本
## 功能
不影响网页端的操作下,绕开流量限制,可以直接点击下载
(文件夹打包仍无法不限流量下载,文件夹请使用下载工具下载)
## 安装方法
## 1.通过油叉安装(推荐)
### 下载油猴扩展
进入谷歌浏览器或edge浏览器扩展商店搜索“篡改猴”安装<br>
![image](https://github.com/user-attachments/assets/88ea4392-7f09-432c-a8d4-06f63781a8ed)
### 安装脚本<br>
打开篡改猴,点击获取新脚本:<br>
![image](https://github.com/user-attachments/assets/5a874ebb-2e5d-423b-9200-9addaeaa32a7)
### 进入油叉: 或直接进入脚本地址:```https://greasyfork.org/zh-CN/scripts/510621-123%E4%BA%91%E7%9B%98%E4%B8%8B%E8%BD%BD%E8%BE%85%E5%8A%A9```<br>
![image](https://github.com/user-attachments/assets/05a1cd0a-ef40-45db-9367-3136e4b7262c)
### 搜索```123云盘下载辅助```:<br>
![image](https://github.com/user-attachments/assets/6b8a5d08-6e8b-404a-a993-8474f85cdb04)<br>
### 安装:<br>
![image](https://github.com/user-attachments/assets/42cc5908-e29e-4ed8-852e-83ee59540bfb)<br>
<br>
![image](https://github.com/user-attachments/assets/3d1c07e6-19dd-4474-bde3-121b493803b8)<br>
![image](https://github.com/user-attachments/assets/788454f3-e8a6-4597-9c67-9a585478dbe7)
### 安装完成后即可进入123云盘官网不限速下载
![image](https://github.com/user-attachments/assets/0a83df5c-c95d-4fa7-804d-e714b78c5a14)
## 2.直接安装
下载本仓库的123download.js安装进油猴即可具体方法不再详述请自行搜索。
# 注意事项
- 请确保在使用过程中网络连接正常。
- 由于使用了模拟安卓客户端协议,可能会有一定的风险,请谨慎使用。
- 本工具仅供学习和研究使用,请勿用于非法用途,任何滥用行为造成的后果由使用者承担

1088
android.py

File diff suppressed because it is too large Load Diff

View File

@ -1,2 +1 @@
pyinstaller -F android.py pyinstaller -F 123pan.py --icon icon.ico --clean --noconfirm
pyinstaller -F web.py

View File

@ -1,3 +1,4 @@
# 网页端加签算法目前123pan已弃用
import time import time
import random import random
from datetime import datetime from datetime import datetime

824
web.py
View File

@ -1,824 +0,0 @@
import re
import time
from sign_py import getSign
import requests
import hashlib
import os
import json
import base64
class Pan123:
def __init__(
self,
readfile=True,
user_name="",
pass_word="",
authorization="",
input_pwd=True,
):
self.recycle_list = None
self.list = None
if readfile:
self.read_ini(user_name, pass_word, input_pwd, authorization)
else:
if user_name == "" or pass_word == "":
print("读取已禁用,用户名或密码为空")
if input_pwd:
user_name = input("请输入用户名:")
pass_word = input("请输入密码:")
else:
raise Exception("用户名或密码为空读取禁用时userName和passWord不能为空")
self.user_name = user_name
self.password = pass_word
self.authorization = authorization
self.header_only_usage = {
"user-agent": "Mozilla/5.0 (Windows NT 10.0) AppleWebKit/"
"537.36 (KHTML, like Gecko) Chrome/109.0.0.0 "
"Safari/537.36 Edg/109.0.1474.0",
"app-version": "2",
"platform": "web",
}
self.header_logined = {
"Accept": "*/*",
"Accept-Language": "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6",
"App-Version": "3",
"Authorization": self.authorization,
"Cache-Control": "no-cache",
"Connection": "keep-alive",
"LoginUuid": "z-uk_yT8HwR4raGX1gqGk",
"Pragma": "no-cache",
"Referer": "https://www.123pan.com/",
"Sec-Fetch-Dest": "empty",
"Sec-Fetch-Mode": "cors",
"Sec-Fetch-Site": "same-origin",
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/"
"537.36 (KHTML, like Gecko) Chrome/119.0.0.0 "
"Safari/537.36 Edg/119.0.0.0",
"platform": "web",
"sec-ch-ua": "^\\^Microsoft",
"sec-ch-ua-mobile": "?0",
"sec-ch-ua-platform": "^\\^Windows^^",
}
self.parent_file_id = 0 # 路径文件夹的id,0为根目录
self.parent_file_list = [0]
res_code_getdir = self.get_dir()
if res_code_getdir != 0:
self.login()
self.get_dir()
def login(self):
data = {"remember": True, "passport": self.user_name, "password": self.password}
sign = getSign("/b/api/user/sign_in")
login_res = requests.post(
"https://www.123pan.com/b/api/user/sign_in",
headers=self.header_only_usage,
data=data,
params={sign[0]: sign[1]}, timeout=10
)
res_sign = login_res.json()
res_code_login = res_sign["code"]
if res_code_login != 200:
print("code = 1 Error:" + str(res_code_login))
print(res_sign["message"])
return res_code_login
token = res_sign["data"]["token"]
self.authorization = "Bearer " + token
header_logined = {
"Accept": "*/*",
"Accept-Language": "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6",
"App-Version": "3",
"Authorization": self.authorization,
"Cache-Control": "no-cache",
"Connection": "keep-alive",
"LoginUuid": "z-uk_yT8HwR4raGX1gqGk",
"Pragma": "no-cache",
"Referer": "https://www.123pan.com/",
"Sec-Fetch-Dest": "empty",
"Sec-Fetch-Mode": "cors",
"Sec-Fetch-Site": "same-origin",
"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/"
"537.36 (KHTML, like"
" Gecko) Chrome/119.0.0.0 Safari/537.36 Edg/119.0.0.0",
"platform": "web",
"sec-ch-ua": "^\\^Microsoft",
"sec-ch-ua-mobile": "?0",
"sec-ch-ua-platform": "^\\^Windows^^",
}
self.header_logined = header_logined
# ret['cookie'] = cookie
self.save_file()
return res_code_login
def save_file(self):
with open("123pan.txt", "w",encoding="utf_8") as f:
save_list = {
"userName": self.user_name,
"passWord": self.password,
"authorization": self.authorization,
}
f.write(json.dumps(save_list))
print("Save!")
def get_dir(self):
res_code_getdir = 0
page = 1
lists = []
lenth_now = 0
total = -1
while lenth_now < total or total == -1:
base_url = "https://www.123pan.com/b/api/file/list/new"
# print(self.headerLogined)
sign = getSign("/b/api/file/list/new")
print(sign)
params = {
sign[0]: sign[1],
"driveId": 0,
"limit": 100,
"next": 0,
"orderBy": "file_id",
"orderDirection": "desc",
"parentFileId": str(self.parent_file_id),
"trashed": False,
"SearchData": "",
"Page": str(page),
"OnlyLookAbnormalFile": 0,
}
a = requests.get(base_url, headers=self.header_logined, params=params, timeout=10)
# print(a.text)
# print(a.headers)
text = a.json()
res_code_getdir = text["code"]
if res_code_getdir != 0:
print(a.text)
print(a.headers)
print("code = 2 Error:" + str(res_code_getdir))
return res_code_getdir
lists_page = text["data"]["InfoList"]
lists += lists_page
total = text["data"]["Total"]
lenth_now += len(lists_page)
page += 1
file_num = 0
for i in lists:
i["FileNum"] = file_num
file_num += 1
self.list = lists
return res_code_getdir
def show(self):
print("--------------------")
for i in self.list:
file_size = i["Size"]
if file_size > 1048576:
download_size_print = str(round(file_size / 1048576, 2)) + "M"
else:
download_size_print = str(round(file_size / 1024, 2)) + "K"
if i["Type"] == 0:
print(
"\033[33m" + "编号:",
self.list.index(i) + 1,
"\033[0m \t\t" + download_size_print + "\t\t\033[36m",
i["FileName"],
"\033[0m",
)
elif i["Type"] == 1:
print(
"\033[35m" + "编号:",
self.list.index(i) + 1,
" \t\t\033[36m",
i["FileName"],
"\033[0m",
)
print("--------------------")
# fileNumber 从0开始0为第一个文件传入时需要减一
def link(self, file_number, showlink=True):
file_detail = self.list[file_number]
type_detail = file_detail["Type"]
if type_detail == 1:
down_request_url = "https://www.123pan.com/a/api/file/batch_download_info"
down_request_data = {"fileIdList": [{"fileId": int(file_detail["FileId"])}]}
else:
down_request_url = "https://www.123pan.com/a/api/file/download_info"
down_request_data = {
"driveId": 0,
"etag": file_detail["Etag"],
"fileId": file_detail["FileId"],
"s3keyFlag": file_detail["S3KeyFlag"],
"type": file_detail["Type"],
"fileName": file_detail["FileName"],
"size": file_detail["Size"],
}
# print(down_request_data)
sign = getSign("/a/api/file/download_info")
link_res = requests.post(
down_request_url,
headers=self.header_logined,
params={sign[0]: sign[1]},
data=down_request_data,
timeout=10
)
# print(linkRes.text)
res_code_download = link_res.json()["code"]
if res_code_download != 0:
print("code = 3 Error:" + str(res_code_download))
# print(linkRes.json())
return res_code_download
download_link_base64 = link_res.json()["data"]["DownloadUrl"]
base64_url = re.findall("params=(.*)&", download_link_base64)[0]
# print(Base64Url)
down_load_url = base64.b64decode(base64_url)
down_load_url = down_load_url.decode("utf-8")
next_to_get = requests.get(down_load_url,timeout=10).json()
redirect_url = next_to_get["data"]["redirect_url"]
if showlink:
print(redirect_url)
return redirect_url
def download(self, file_number):
file_detail = self.list[file_number]
down_load_url = self.link(file_number, showlink=False)
file_name = file_detail["FileName"] # 文件名
if os.path.exists(file_name):
print("文件 " + file_name + " 已存在,是否要覆盖?")
sure_download = input("输入1覆盖2取消")
if sure_download != "1":
return
down = requests.get(down_load_url, stream=True, timeout=10)
file_size = int(down.headers["Content-Length"]) # 文件大小
content_size = int(file_size) # 文件总大小
data_count = 0 # 当前已传输的大小
if file_size > 1048576:
size_print_download = str(round(file_size / 1048576, 2)) + "M"
else:
size_print_download = str(round(file_size / 1024, 2)) + "K"
print(file_name + " " + size_print_download)
time1 = time.time()
time_temp = time1
data_count_temp = 0
with open(file_name, "wb") as f:
for i in down.iter_content(1024):
f.write(i)
done_block = int((data_count / content_size) * 50)
data_count = data_count + len(i)
# 实时进度条进度
now_jd = (data_count / content_size) * 100
# %% 表示%
# 测速
time1 = time.time()
pass_time = time1 - time_temp
if pass_time > 1:
time_temp = time1
pass_data = int(data_count) - int(data_count_temp)
data_count_temp = data_count
speed = pass_data / int(pass_time)
speed_m = speed / 1048576
if speed_m > 1:
speed_print = str(round(speed_m, 2)) + "M/S"
else:
speed_print = str(round(speed_m * 1024, 2)) + "K/S"
print(
"\r [%s%s] %d%% %s"
% (
done_block * "",
" " * (50 - 1 - done_block),
now_jd,
speed_print,
),
end="",
)
elif data_count == content_size:
print("\r [%s%s] %d%% %s" % (50 * "", "", 100, ""), end="")
print("\nok")
def recycle(self):
recycle_id = 0
url = (
"https://www.123pan.com/a/api/file/list/new?driveId=0&limit=100&next=0"
"&orderBy=fileId&orderDirection=desc&parentFileId="
+ str(recycle_id)
+ "&trashed=true&&Page=1"
)
recycle_res = requests.get(url, headers=self.header_logined, timeout=10)
json_recycle = recycle_res.json()
recycle_list = json_recycle["data"]["InfoList"]
self.recycle_list = recycle_list
# fileNumber 从0开始0为第一个文件传入时需要减一
def delete_file(self, file, by_num=True, operation=True):
# operation = 'true' 删除 operation = 'false' 恢复
if by_num:
print(file)
if not str(file).isdigit():
print("请输入数字")
return -1
if 0 <= file < len(self.list):
file_detail = self.list[file]
else:
print("不在合理范围内")
return
else:
if file in self.list:
file_detail = file
else:
print("文件不存在")
return
data_delete = {
"driveId": 0,
"fileTrashInfoList": file_detail,
"operation": operation,
}
delete_res = requests.post(
"https://www.123pan.com/a/api/file/trash",
data=json.dumps(data_delete),
headers=self.header_logined,
timeout=10
)
dele_json = delete_res.json()
print(dele_json)
message = dele_json["message"]
print(message)
def share(self):
file_id_list = ""
share_name_list = []
add = "1"
while str(add) == "1":
share_num = input("分享文件的编号:")
num_test2 = share_num.isdigit()
if num_test2:
share_num = int(share_num)
if 0 < share_num < len(self.list) + 1:
share_id = self.list[int(share_num) - 1]["FileId"]
share_name = self.list[int(share_num) - 1]["FileName"]
share_name_list.append(share_name)
print(share_name_list)
file_id_list = file_id_list + str(share_id) + ","
add = input("输入1添加文件0发起分享其他取消")
else:
print("请输入数字,,")
add = "1"
if str(add) == "0":
share_pwd = input("提取码,不设留空:")
file_id_list = file_id_list.strip(",")
data = {
"driveId": 0,
"expiration": "2024-02-09T11:42:45+08:00",
"fileIdList": file_id_list,
"shareName": "我的分享",
"sharePwd": share_pwd,
}
share_res = requests.post(
"https://www.123pan.com/a/api/share/create",
headers=self.header_logined,
data=json.dumps(data),
timeout=10
)
share_res_json = share_res.json()
message = share_res_json["message"]
print(message)
share_key = share_res_json["data"]["ShareKey"]
share_url = "https://www.123pan.com/s/" + share_key
print("分享链接:\n" + share_url + "提取码:" + share_pwd)
else:
print("退出分享")
def up_load(self, file_path):
file_path = file_path.replace('"', "")
file_path = file_path.replace("\\", "/")
file_name = file_path.split("/")[-1]
print("文件名:", file_name)
if not os.path.exists(file_path):
print("文件不存在,请检查路径是否正确")
return
if os.path.isdir(file_path):
print("暂不支持文件夹上传")
return
fsize = os.path.getsize(file_path)
with open(file_path, "rb") as f:
md5 = hashlib.md5()
while True:
data = f.read(64 * 1024)
if not data:
break
md5.update(data)
readable_hash = md5.hexdigest()
list_up_request = {
"driveId": 0,
"etag": readable_hash,
"fileName": file_name,
"parentFileId": self.parent_file_id,
"size": fsize,
"type": 0,
"duplicate": 0,
}
sign = getSign("/b/api/file/upload_request")
up_res = requests.post(
"https://www.123pan.com/b/api/file/upload_request",
headers=self.header_logined,
params={sign[0]: sign[1]},
data=list_up_request,
timeout=10
)
up_res_json = up_res.json()
res_code_up = up_res_json["code"]
if res_code_up == 5060:
sure_upload = input("检测到1个同名文件,输入1覆盖2保留两者0取消")
if sure_upload == "1":
list_up_request["duplicate"] = 1
elif sure_upload == "2":
list_up_request["duplicate"] = 2
else:
print("取消上传")
return
sign = getSign("/b/api/file/upload_request")
up_res = requests.post(
"https://www.123pan.com/b/api/file/upload_request",
headers=self.header_logined,
params={sign[0]: sign[1]},
data=json.dumps(list_up_request),
timeout=10
)
up_res_json = up_res.json()
res_code_up = up_res_json["code"]
if res_code_up == 0:
# print(upResJson)
# print("上传请求成功")
reuse = up_res_json["data"]["Reuse"]
if reuse:
print("上传成功文件已MD5复用")
return
else:
print(up_res_json)
print("上传请求失败")
return
bucket = up_res_json["data"]["Bucket"]
storage_node = up_res_json["data"]["StorageNode"]
upload_key = up_res_json["data"]["Key"]
upload_id = up_res_json["data"]["UploadId"]
up_file_id = up_res_json["data"]["FileId"] # 上传文件的fileId,完成上传后需要用到
print("上传文件的fileId:", up_file_id)
# 获取已将上传的分块
start_data = {
"bucket": bucket,
"key": upload_key,
"uploadId": upload_id,
"storageNode": storage_node,
}
start_res = requests.post(
"https://www.123pan.com/b/api/file/s3_list_upload_parts",
headers=self.header_logined,
data=json.dumps(start_data),
timeout=10
)
start_res_json = start_res.json()
res_code_up = start_res_json["code"]
if res_code_up == 0:
# print(startResJson)
pass
else:
print(start_data)
print(start_res_json)
print("获取传输列表失败")
return
# 分块,每一块取一次链接,依次上传
block_size = 5242880
with open(file_path, "rb") as f:
part_number_start = 1
put_size = 0
while True:
data = f.read(block_size)
precent = round(put_size / fsize, 2)
print("\r已上传:" + str(precent * 100) + "%", end="")
put_size = put_size + len(data)
if not data:
break
get_link_data = {
"bucket": bucket,
"key": upload_key,
"partNumberEnd": part_number_start + 1,
"partNumberStart": part_number_start,
"uploadId": upload_id,
"StorageNode": storage_node,
}
get_link_url = (
"https://www.123pan.com/b/api/file/s3_repare_upload_parts_batch"
)
get_link_res = requests.post(
get_link_url,
headers=self.header_logined,
data=json.dumps(get_link_data),
timeout=10
)
get_link_res_json = get_link_res.json()
res_code_up = get_link_res_json["code"]
if res_code_up == 0:
# print("获取链接成功")
pass
else:
print("获取链接失败")
# print(getLinkResJson)
return
# print(getLinkResJson)
upload_url = get_link_res_json["data"]["presignedUrls"][
str(part_number_start)
]
# print("上传链接",uploadUrl)
requests.put(upload_url, data=data, timeout=10)
# print("put")
part_number_start = part_number_start + 1
print("\n处理中")
# 完成标志
# 1.获取已上传的块
uploaded_list_url = "https://www.123pan.com/b/api/file/s3_list_upload_parts"
uploaded_comp_data = {
"bucket": bucket,
"key": upload_key,
"uploadId": upload_id,
"storageNode": storage_node,
}
# print(uploadedCompData)
requests.post(
uploaded_list_url,
headers=self.header_logined,
data=json.dumps(uploaded_comp_data),
timeout=10
)
compmultipart_up_url = (
"https://www.123pan.com/b/api/file/s3_complete_multipart_upload"
)
requests.post(
compmultipart_up_url,
headers=self.header_logined,
data=json.dumps(uploaded_comp_data),
timeout=10
)
# 3.报告完成上传关闭upload session
if fsize > 64 * 1024 * 1024:
time.sleep(3)
close_up_session_url = "https://www.123pan.com/b/api/file/upload_complete"
close_up_session_data = {"fileId": up_file_id}
# print(closeUpSessionData)
close_up_session_res = requests.post(
close_up_session_url,
headers=self.header_logined,
data=json.dumps(close_up_session_data),
timeout=10
)
close_res_json = close_up_session_res.json()
# print(closeResJson)
res_code_up = close_res_json["code"]
if res_code_up == 0:
print("上传成功")
else:
print("上传失败")
print(close_res_json)
return
# dirId 就是 fileNumber从0开始0为第一个文件传入时需要减一 !!!(好像文件夹都排在前面)
def cd(self, dir_num):
if not dir_num.isdigit():
if dir_num == "..":
if len(self.parent_file_list) > 1:
self.parent_file_list.pop()
self.parent_file_id = self.parent_file_list[-1]
self.get_dir()
self.show()
else:
print("已经是根目录")
return
if dir_num == "/":
self.parent_file_id = 0
self.parent_file_list = [0]
self.get_dir()
self.show()
return
print("输入错误")
return
dir_num = int(dir_num) - 1
if dir_num >= (len(self.list) - 1) or dir_num < 0:
print("输入错误")
return
if self.list[dir_num]["Type"] != 1:
print("不是文件夹")
return
self.parent_file_id = self.list[dir_num]["FileId"]
self.parent_file_list.append(self.parent_file_id)
self.get_dir()
self.show()
def cdById(self, file_id):
self.parent_file_id = file_id
self.parent_file_list.append(self.parent_file_id)
self.get_dir()
self.get_dir()
self.show()
def read_ini(
self,
user_name,
pass_word,
input_pwd,
authorization="",
):
try:
with open("123pan.txt", "r",encoding="utf-8") as f:
text = f.read()
text = json.loads(text)
user_name = text["userName"]
pass_word = text["passWord"]
authorization = text["authorization"]
except:
print("获取配置失败,重新登录")
if user_name == "" or pass_word == "":
if input_pwd:
user_name = input("userName:")
pass_word = input("passWord:")
authorization = ""
else:
raise Exception("禁止输入模式下,没有账号或密码")
self.user_name = user_name
self.password = pass_word
self.authorization = authorization
def mkdir(self, dirname, remakedir=False):
if not remakedir:
for i in self.list:
if i["FileName"] == dirname:
print("文件夹已存在")
return i["FileId"]
url = "https://www.123pan.com/a/api/file/upload_request"
data_mk = {
"driveId": 0,
"etag": "",
"fileName": dirname,
"parentFileId": self.parent_file_id,
"size": 0,
"type": 1,
"duplicate": 1,
"NotReuse": True,
"event": "newCreateFolder",
"operateType": 1,
}
sign = getSign("/a/api/file/upload_request")
res_mk = requests.post(
url,
headers=self.header_logined,
data=json.dumps(data_mk),
params={sign[0]: sign[1]},
timeout=10
)
try:
res_json = res_mk.json()
print(res_json)
except json.decoder.JSONDecodeError:
print("创建失败")
print(res_mk.text)
return
code_mkdir = res_json["code"]
if code_mkdir == 0:
print("创建成功: ", res_json["data"]["FileId"])
self.get_dir()
return res_json["data"]["Info"]["FileId"]
print("创建失败")
print(res_json)
return
if __name__ == "__main__":
print("web协议将废弃请使用android协议")
pan = Pan123(readfile=True, input_pwd=True)
pan.show()
while True:
command = input("\033[91m >\033[0m")
if command == "ls":
pan.show()
if command == "re":
code = pan.get_dir()
if code == 0:
print("刷新目录成功")
pan.show()
if command.isdigit():
if int(command) > len(pan.list) or int(command) < 1:
print("输入错误")
continue
if pan.list[int(command) - 1]["Type"] == 1:
pan.cdById(pan.list[int(command) - 1]["FileId"])
else:
size = pan.list[int(command) - 1]["Size"]
if size > 1048576:
size_print_show = str(round(size / 1048576, 2)) + "M"
else:
size_print_show = str(round(size / 1024, 2)) + "K"
# print(pan.list[int(command) - 1])
name = pan.list[int(command) - 1]["FileName"]
print(name + " " + size_print_show)
print("press 1 to download now: ", end="")
sure = input()
if sure == "1":
pan.download(int(command) - 1)
elif command[0:9] == "download ":
if command[9:].isdigit():
if int(command[9:]) > len(pan.list) or int(command[9:]) < 1:
print("输入错误")
continue
pan.download(int(command[9:]) - 1)
else:
print("输入错误")
elif command == "exit":
break
elif command == "log":
pan.login()
pan.get_dir()
pan.show()
elif command[0:5] == "link ":
if command[5:].isdigit():
if int(command[5:]) > len(pan.list) or int(command[5:]) < 1:
print("输入错误")
continue
pan.link(int(command[5:]) - 1)
else:
print("输入错误")
elif command == "upload":
filepath = input("请输入文件路径:")
pan.up_load(filepath)
pan.get_dir()
pan.show()
elif command == "share":
pan.share()
elif command[0:6] == "delete":
if command == "delete":
print("请输入要删除的文件编号:", end="")
fileNumber = input()
else:
if command[6] == " ":
fileNumber = command[7:]
else:
print("输入错误")
continue
if fileNumber == "":
print("请输入要删除的文件编号:", end="")
fileNumber = input()
else:
fileNumber = fileNumber[0:]
if fileNumber.isdigit():
if int(fileNumber) > len(pan.list) or int(fileNumber) < 1:
print("输入错误")
continue
pan.delete_file(int(fileNumber) - 1)
pan.get_dir()
pan.show()
else:
print("输入错误")
elif command[:3] == "cd ":
path = command[3:]
pan.cd(path)
elif command[0:5] == "mkdir":
if command == "mkdir":
newPath = input("请输入目录名:")
else:
newPath = command[6:]
if newPath == "":
newPath = input("请输入目录名:")
else:
newPath = newPath[0:]
print(pan.mkdir(newPath))
elif command == "reload":
pan.read_ini("", "", True)
print("读取成功")
pan.get_dir()
pan.show()