青龙面板教程:天翼云盘青龙签到并推送通知手机

前言

电信天翼云盘是真不错,是真正的不限速。可惜空间太小了,但是可以通过签到来获取永久容量,虽然一次签到只有几十M,但是写个脚本一直挂着也不错,积少成多。下面把折腾方法分享出来。

效果演示:

部署方法:

  • 1、进青龙面板-脚本管理-新增脚本tianyiyun.py,如下图:
  • 2、进环境变量-创建变量。PUSHDEER_PUSHKEY,ty_username和ty_password,分别对应填写pushdeer密钥,天翼云账号和密码。如下图:

  • 3、定时任务里面创建任务,按下图所示填写即可。

  • 手动运行一次,成功即可。

教程扩展一下,如果出现设备锁问题解决方法:

最新:关于天翼云设备锁问题导致登陆失败或者提示登录错误:设备ID不存在,需要二次设备校验的解决办法!

  • 1、登陆这个网址:https://e.dlife.cn/user/index.do,会弹出新网页,默认显示【个人信息】,点左侧【帐号安全】
  • 2、关闭【设备锁】,即可
  • Ps:网页关闭不了设备锁的,用 天翼云盘 app关闭自行寻找 期间需要两次验证码验证! 如果对上述网址不放心的可以用下面的方法 1、登录天翼云盘网页版,地址:https://cloud.189.cn/web/login.html 2、登录后,鼠标移动到右上方帐号位置,并点击【帐号设置】
  • 3、点【更改头像】
  • 4、会弹出新网页,默认显示【个人信息】,点左侧【帐号安全】
  • 5、关闭【设备锁】,即可
  • 6、测试截图如下,左侧为未关闭【设备锁】,右侧为关闭【设备锁】之后 ps 其实你点击头像后就是这个网址 只不过我直接提取了 期间需要两次验证码验证!

最后附上天翼云脚本:

import time
import os
import random
import json
import base64
import hashlib
import rsa
import requests
import re
from urllib.parse import urlparse
BI_RM = list("0123456789abcdefghijklmnopqrstuvwxyz")
B64MAP = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"
# 从环境变量获取账号信息
ty_usernames = os.getenv("ty_username").split('&') if os.getenv("ty_username") else []
ty_passwords = os.getenv("ty_password").split('&') if os.getenv("ty_password") else []
# 检查环境变量
if not ty_usernames or not ty_passwords:
    raise ValueError("❌ 请设置环境变量 ty_username 和 ty_password")
# 组合账号信息
accounts = [{"username": u, "password": p} for u, p in zip(ty_usernames, ty_passwords)]
# PushDeer配置
PUSHDEER_PUSHKEYS = os.getenv("PUSHDEER_PUSHKEY", "").split('&')
def mask_phone(phone):
    """隐藏手机号中间四位"""
    return phone[:3] + "****" + phone[-4:] if len(phone) == 11 else phone[:3] + "****" + phone[-4:]
def int2char(a):
    return BI_RM[a]
def b64tohex(a):
    d = ""
    e = 0
    c = 0
    for i in range(len(a)):
        if list(a)[i] != "=":
            v = B64MAP.index(list(a)[i])
            if 0 == e:
                e = 1
                d += int2char(v >> 2)
                c = 3 & v
            elif 1 == e:
                e = 2
                d += int2char(c << 2 | v >> 4)
                c = 15 & v
            elif 2 == e:
                e = 3
                d += int2char(c)
                d += int2char(v >> 2)
                c = 3 & v
            else:
                e = 0
                d += int2char(c << 2 | v >> 4)
                d += int2char(15 & v)
    if e == 1:
        d += int2char(c << 2)
    return d
def rsa_encode(j_rsakey, string):
    rsa_key = f"-----BEGIN PUBLIC KEY-----\n{j_rsakey}\n-----END PUBLIC KEY-----"
    pubkey = rsa.PublicKey.load_pkcs1_openssl_pem(rsa_key.encode())
    result = b64tohex((base64.b64encode(rsa.encrypt(f'{string}'.encode(), pubkey))).decode())
    return result
def login(username, password):
    print("🔄 正在执行登录流程...")
    s = requests.Session()
    try:
        urlToken = "https://m.cloud.189.cn/udb/udb_login.jsp?pageId=1&pageKey=default&clientType=wap&redirectURL=https://m.cloud.189.cn/zhuanti/2021/shakeLottery/index.html"
        r = s.get(urlToken)
        match = re.search(r"https?://[^\s'\"]+", r.text)
        if not match:
            print("❌ 错误:未找到动态登录页")
            return None

        url = match.group()
        r = s.get(url)
        match = re.search(r"<a id=\"j-tab-login-link\"[^>]*href=\"([^\"]+)\"", r.text)
        if not match:
            print("❌ 错误:登录入口获取失败")
            return None

        href = match.group(1)
        r = s.get(href)

        captchaToken = re.findall(r"captchaToken' value='(.+?)'", r.text)[0]
        lt = re.findall(r'lt = "(.+?)"', r.text)[0]
        returnUrl = re.findall(r"returnUrl= '(.+?)'", r.text)[0]
        paramId = re.findall(r'paramId = "(.+?)"', r.text)[0]
        j_rsakey = re.findall(r'j_rsaKey" value="(\S+)"', r.text, re.M)[0]
        s.headers.update({"lt": lt})
        username_enc = rsa_encode(j_rsakey, username)
        password_enc = rsa_encode(j_rsakey, password)

        data = {
            "appKey": "cloud",
            "accountType": '01',
            "userName": f"{{RSA}}{username_enc}",
            "password": f"{{RSA}}{password_enc}",
            "validateCode": "",
            "captchaToken": captchaToken,
            "returnUrl": returnUrl,
            "mailSuffix": "@189.cn",
            "paramId": paramId
        }

        headers = {
            'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:74.0) Gecko/20100101 Firefox/76.0',
            'Referer': 'https://open.e.189.cn/',
        }

        r = s.post(
            "https://open.e.189.cn/api/logbox/oauth2/loginSubmit.do",
            data=data,
            headers=headers,
            timeout=10
        )

        if r.json().get('result', 1) != 0:
            print(f"❌ 登录错误:{r.json().get('msg')}")
            return None

        s.get(r.json()['toUrl'])
        print("✅ 登录成功")
        return s

    except Exception as e:
        print(f"⚠️ 登录异常:{str(e)}")
        return None
def send_pushdeer(msg):
    if not PUSHDEER_PUSHKEYS:
        print("⚠️ 未配置PushDeer,跳过消息推送")
        return

    # PushDeer推送API
    base_url = "https://api2.pushdeer.com/message/push"

    for pushkey in PUSHDEER_PUSHKEYS:
        if not pushkey.strip():
            continue

        data = {
            "pushkey": pushkey.strip(),
            "text": "天翼云盘签到结果",
            "desp": msg,
            "type": "markdown"
        }

        try:
            resp = requests.post(base_url, data=data, timeout=10)
            result = resp.json()
            if result.get('code') == 0:
                print(f"✅ PushDeer消息推送成功 -> PushKey: {pushkey[:4]}...")
            else:
                print(f"❌ PushDeer消息推送失败:{result.get('error', '未知错误')}")
        except Exception as e:
            print(f"❌ PushDeer推送异常:{str(e)}")
def main():
    print("\n=============== 天翼云盘签到开始 ===============")
    all_results = []

    for acc in accounts:
        username = acc["username"]
        password = acc["password"]
        masked_phone = mask_phone(username)
        account_result = {"username": masked_phone, "sign": "", "lottery": ""}

        print(f"\n🔔 处理账号:{masked_phone}")

        # 登录流程
        session = login(username, password)
        if not session:
            account_result["sign"] = "❌ 登录失败"
            all_results.append(account_result)
            continue

        # 签到流程
        try:
            # 每日签到
            rand = str(round(time.time() * 1000))
            sign_url = f'https://api.cloud.189.cn/mkt/userSign.action?rand={rand}&clientType=TELEANDROID&version=8.6.3&model=SM-G930K'
            headers = {
                'User-Agent': 'Mozilla/5.0 (Linux; Android 5.1.1; SM-G930K Build/NRD90M; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/74.0.3729.136 Mobile Safari/537.36 Ecloud/8.6.3 Android/22 clientId/355325117317828 clientModel/SM-G930K imsi/460071114317824 clientChannelId/qq proVersion/1.0.6',
                "Referer": "https://m.cloud.189.cn/zhuanti/2016/sign/index.jsp?albumBackupOpened=1",
                "Host": "m.cloud.189.cn",
            }
            resp = session.get(sign_url, headers=headers).json()
            if resp.get('isSign') == "false":
                account_result["sign"] = f"✅ +{resp['netdiskBonus']}M"
            else:
                account_result["sign"] = f"⏳ 已签到+{resp['netdiskBonus']}M"

            # 单次抽奖(原第一次抽奖)
            time.sleep(random.randint(2, 5))
            lottery_url = 'https://m.cloud.189.cn/v2/drawPrizeMarketDetails.action?taskId=TASK_SIGNIN&activityId=ACT_SIGNIN'
            resp = session.get(lottery_url, headers=headers).json()
            if "errorCode" in resp:
                account_result["lottery"] = f"❌ {resp.get('errorCode')}"
            else:
                prize = resp.get('prizeName') or resp.get('description')
                account_result["lottery"] = f"🎁 {prize}"

        except Exception as e:
            account_result["sign"] = "❌ 操作异常"
            account_result["lottery"] = f"⚠️ {str(e)}"

        all_results.append(account_result)
        print(f"  {account_result['sign']} | {account_result['lottery']}")

    # 生成汇总表格
    table = "### ⛅ 天翼云盘签到汇总\n\n"
    table += "| 账号 | 签到结果 | 每日抽奖 |\n"
    table += "|:-:|:-:|:-:|\n"
    for res in all_results:
        table += f"| {res['username']} | {res['sign']} | {res['lottery']} |\n"

    # 发送汇总推送
    send_pushdeer(table)
    print("\n✅ 所有账号处理完成!")
if __name__ == "__main__":
    main()