# !/usr/bin/python3 # -- coding: utf-8 -- # ------------------------------- # @Author : cherwin # ✨✨✨ 韵达快递小程序签到✨✨✨ # ✨ 功能: # 积分签到 # ✨ 抓包步骤: # 打开韵达快递小程序 # 授权登陆 # 打开抓包工具 # 找URl请求头带openID # 复制里面的openID参数值 # 参数示例:at965eoakumxxxxxxxx # ✨ 设置青龙变量: # export YDKD='openID参数值'多账号#或&分割 #=======0802 加了一个ocrserver 修复抽奖 ==== # export SCRIPT_UPDATE = 'False' 关闭脚本自动更新,默认开启 # ✨ ✨ 注意:抓完CK没事儿别打开小程序,重新打开小程序请重新抓包 # ✨ 推荐cron:0 6 * * * # ✨✨✨ @Author CHERWIN✨✨✨ # ------------------------------- # cron "30 6 * * *" script-path=xxx.py,tag=匹配cron用 # const $ = new Env('韵达快递小程序签到') import json import os import random import time from os import path import requests import hashlib from requests.packages.urllib3.exceptions import InsecureRequestWarning # import CHERWIN_TOOLS # 禁用安全请求警告 requests.packages.urllib3.disable_warnings(InsecureRequestWarning) IS_DEV = False if os.path.isfile('DEV_ENV.py'): import DEV_ENV IS_DEV = True if os.path.isfile('notify.py'): from notify import send print("加载通知服务成功!") else: print("加载通知服务失败!") send_msg = '' one_msg='' def Log(cont=''): global send_msg,one_msg print(cont) if cont: one_msg += f'{cont}\n' send_msg += f'{cont}\n' class RUN: def __init__(self,info,index): global one_msg one_msg = '' split_info = info.split('@') self.token = split_info[0] len_split_info = len(split_info) last_info = split_info[len_split_info - 1] self.send_UID = None if len_split_info > 0 and "UID_" in last_info: print('检测到设置了UID') print(last_info) self.send_UID = last_info self.index = index + 1 self.headers = { 'Host': 'op.yundasys.com', 'Accept': 'application/json, text/plain, */*', 'X-Requested-With': 'XMLHttpRequest', 'Authorization': self.token, 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4758.102 Safari/537.36 MicroMessenger/7.0.20.1781(0x6700143B) NetType/WIFI MiniProgramEnv/Windows WindowsWechat/WMPF XWEB/6945', 'Content-Type': 'application/json; charset=UTF-8', 'Sec-Fetch-Site': 'same-origin', 'Sec-Fetch-Mode': 'cors', 'Sec-Fetch-Dest': 'empty', 'Referer': 'https://op.yundasys.com/mb-ext-channel/index.html', 'Accept-Language': 'zh-CN,zh' } self.yzmheaders = { 'Host': 'mbhtml.yundasys.com', 'Authorization': self.token, 'Accept': 'application/json, text/plain, */*', 'X-Requested-With': 'XMLHttpRequest', 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/98.0.4758.102 Safari/537.36 MicroMessenger/7.0.20.1781(0x6700143B) NetType/WIFI MiniProgramEnv/Windows WindowsWechat/WMPF XWEB/6945', 'Content-Type': 'application/json; charset=UTF-8', 'Sec-Fetch-Site': 'same-origin', 'Sec-Fetch-Mode': 'cors', 'Sec-Fetch-Dest': 'empty', 'Referer': 'https://mbhtml.yundasys.com/mb-ext-channel/index.html', 'Accept-Language': 'zh-CN,zh' } self.baseUrl = 'https://op.yundasys.com/gateway/ydmb-integral/ydintegral/' self.news_List = [] self.list_index = 0 self.isFirstTask = True def get_point(self): Log('>>>>>>获取积分信息') json_data = { "channelId": "wxapp", "reqTime": int(time.time()), "accountSrc": "wxapp", "accountId": self.token } response = s.post(f'{self.baseUrl}member/integral/info', headers=self.headers,json=json_data) point_info = response.json() # print(point_info) if point_info['code']== 200: point=point_info['data']['total'] userId=point_info['data']['userId'] Log(f'>>当前用户:【{userId}】') Log(f'>>当前积分:【{point}】') return True else: Log('可能token失效了') return False def get_yzm_x(self,jigsawImageUrl,originalImageUrl): # Log('>>>>>>获取验证码x坐标') # originalImageUrl背景 json_data = {'slidingImage': jigsawImageUrl, 'backImage': originalImageUrl} xheaders={'Content-Type': 'application/json'} try: OCR_SERVER = os.environ.get('OCR_SERVER') response = s.post(f'{OCR_SERVER}/capcode', headers=xheaders,json=json_data) # 检查响应状态码,确保请求成功 response.raise_for_status() # 如果响应状态码不是200系列,将抛出HTTPError异常 # 尝试将响应内容解析为JSON res = response.json() # 从响应的JSON数据中提取'result'字段 x = res.get('result', 0) # 使用.get()可以安全地处理字典中不存在的键,返回None或其他默认值 # 现在你可以安全地使用x了 print(f"验证码x坐标计算成功") except requests.RequestException as e: # 处理所有来自requests的异常,比如连接错误、超时、HTTP错误等 print(f"请求发生错误: {e}") except ValueError as e: # 如果response.json()失败(比如响应不是有效的JSON),则会抛出ValueError print(f"响应不是有效的JSON格式: {e}") except Exception as e: # 处理其他可能的异常 print(f"发生未知错误: {e}") return x def get_yzm(self): # Log('>>>>>>获取验证码信息') json_data = { "client": "mobile", "slideImageWidth": 327, "type": "slide", "reqTime":int(time.time()), "accountId": self.token, "accountSrc": "wxapp" } response = s.post(f'https://mbhtml.yundasys.com/gateway/ydmb-account/ydaccount/getImageVerifyCode', headers=self.yzmheaders,json=json_data) point_info = response.json() # print(f"get_yzm={point_info}") if point_info['code']== 200: # 背景图 shadeImage=point_info['data']['shadeImage'] # 滑块图 cutoutImage=point_info['data']['cutoutImage'] y=point_info['data']['y'] flag=point_info['data']['flag'] # Log(f'>>当前flag:【{flag}】') # Log(f'>>当前y:【{y}】') # 获取x值 x=self.get_yzm_x(cutoutImage,shadeImage) # print(type(x)) # print(type(y)) imageCode=str(x)+"|"+str(y) return flag,imageCode else: Log('可能token失效了') return flag,0 def sign(self): flag,imageCode=self.get_yzm() # Log(f'>>当前flag:【{flag}】') # Log(f'>>当前imageCode:【{imageCode}】') Log('>>>>>>签到') # flag有 imageCode第二个有 json_data = { "itgType": "sign", "channelId": "wxapp", "reqTime": int(time.time()), "accountSrc": "wxapp", "imageCode": imageCode, "flag": flag, "accountId": self.token } response = s.post(f'{self.baseUrl}obtain/event/integral', headers=self.headers,json=json_data) point_info = response.json() # print(point_info) if point_info['code']== 200: msg=point_info['message'] Log(f'>>签到成功,,{msg}') elif point_info['code']== 803: print(f">>1000 {point_info['message']}") print(response.text) else: print(f">>{point_info['message']}") print(response.text) def get_TaskList(self): flag,imageCode=self.get_yzm() Log('>>>>>>获取任务列表') json_data ={ "channelId": "wxapp", "pageNum": 1, "pageSize": 100, "businessType": "goldBetter", "reqTime": int(time.time()), "accountSrc": "wxapp", "imageCode": imageCode, "flag": flag, "accountId": self.token } response = s.post(f'{self.baseUrl}integral/event/list', headers=self.headers,json=json_data) # try: json_data = response.json() # print(json_data) skip_type=['关注公众号','实名认证','完善个人信息','累计消耗积分','寄快递','购买超级会员','兑换商品'] complete = {} if json_data['message'] == '请求成功': data=json_data['data'] list = data['items'] for i in list: eventStatus = i['eventStatus'] eventCode = i['eventCode'] #可完成次数 surplusCount = i['surplusCount'] title = i['eventName'] if title == '本月寄满3件': surplusCount = 1 if title in skip_type:continue stu = {"0":"已完成","1":"未完成"} Log(f'当前任务【{title}】,{stu[eventStatus]}') for t in range(surplusCount): if eventStatus != "1" : self.doTask(eventCode,title) if title == '观看精彩视频': self.watchAd(title) else: complete[title] = 0 time.sleep(2) time.sleep(2) if all(value == 0 for value in complete.values()): Log(">>>任务已全部完成") return True # print(f'>>当前积分:【{point}】') # except: # print(response.text) def watchAd(self,title): flag,imageCode=self.get_yzm() json_data = { "action":"ydmbintegral.ydintegral.obtain.event.integral", "appid":"wjvxmno358lze827", "req_time":int(time.time()), "options":"false", "data":{ "accountId":self.token, "accountSrc":"wxapp", "imageCode": imageCode, "flag": flag, "reqTime":int(time.time()),"itgType":"wechat_viewadv" }, "version":"V1.0"} response = s.post(f'{self.baseUrl}obtain/event/integral', headers=self.headers, json=json_data) point_info = response.json() # print(point_info) if point_info['code'] == 200: msg = point_info['message'] print(f'>>{title},{msg}') else: print(f">>{title},{point_info['message']}") def doTask(self,eventCode,title): flag,imageCode=self.get_yzm() json_data = { 'channelId': 'wxapp', 'itgType': f'{eventCode}', 'reqTime': int(time.time()), 'accountSrc': 'wxapp', "imageCode": imageCode, "flag": flag, 'accountId': self.token } response = s.post(f'{self.baseUrl}obtain/event/integral', headers=self.headers, json=json_data) point_info = response.json() # print(point_info) if point_info['code'] == 200: msg = point_info['message'] print(f'>>{title},{msg}') else: print(f">>{title},{point_info['message']}") def getDrawInfo(self): self.DrawHeaders = { # 'Host': 'op.yundasys.com', 'Host': 'mbhtml.yundasys.com', 'Accept': 'application/json, text/plain, */*', 'X-Requested-With': 'XMLHttpRequest', 'Authorization': self.token, 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/107.0.0.0 Safari/537.36 MicroMessenger/7.0.20.1781(0x6700143B) NetType/WIFI MiniProgramEnv/Windows WindowsWechat/WMPF XWEB/8259', 'Content-Type': 'application/json; charset=UTF-8', 'Sec-Fetch-Site': 'same-origin', 'Sec-Fetch-Mode': 'cors', 'Sec-Fetch-Dest': 'empty', # 'Referer': 'https://op.yundasys.com/mb-ext-channel/index.html', 'Referer': 'https://mbhtml.yundasys.com/mb-ext-channel/index.html', 'Accept-Language': 'zh-CN,zh', } json_data = { "reqTime": int(time.time()), "accountId": self.token, "accountSrc": "wxapp" } # response = s.post(f'https://op.yundasys.com/gateway/ydmbaccount/ydaccount/mc/Itg/store/token', headers=self.DrawHeaders, json=json_data) response = s.post(f'https://mbhtml.yundasys.com/gateway/ydmbaccount/ydaccount/mc/Itg/store/token', headers=self.DrawHeaders, json=json_data) point_info = response.json() print(f"point_info={point_info}") if point_info['code'] == 200: data = point_info['data'] if data: self.getDrawNumber(data) else: print(f">>{point_info['message']}") def getDrawNumber(self,data): json_data = { 'activityId': 16, 'plum_session_applet': data, 'suid': 'gmrtxvrye6', 'mwl_client_flag': 'wxapp', } # resp = s.post(f'https://op.yundasys.com/itgstoresys/api/lottery/drawNumber', headers=self.DrawHeaders, json=json_data) resp = s.post(f'https://mbhtml.yundasys.com/itgstoresys/api/lottery/drawNumber', headers=self.DrawHeaders, json=json_data) res_info = resp.json() if res_info['code'] == 200: freeDrawNumber = res_info['data']['freeDrawNumber'] print(f'>>剩余免费抽奖次数:【{freeDrawNumber}】') print(f"getDrawNumber data={data}") if freeDrawNumber == 1: self.doDraw(data) else: print(f">>{res_info['message']}") def doDraw(self,data): # print(f"doDraw data={data}") flag,imageCode=self.get_yzm() # Log(f'>>当前flag:【{flag}】') # Log(f'>>当前imageCode:【{imageCode}】') res_data = { 'activityId': 16, 'plumSessionApplet': data, 'suid': 'gmrtxvrye6', 'mwlClientFlag': 'wxapp', "accountSrc": "wxapp", "imageCode": imageCode, "flag": flag, "reqTime": int(time.time()), "accountId": self.token } # { # "activityId": 16, # "flag": "d d", # "imageCode": "105|79", # "plumSessionApplet": "6 9", # "suid": "gmrtxvrye6", # "mwlClientFlag": "wxapp", # "reqTime": 17 2, # "accountSrc": "wxapp", # "accountId": "o " # } # resp = s.post(f'https://op.yundasys.com/itgstoresys/api/lottery/draw', headers=self.DrawHeaders, json=res_data) resp = s.post(f'https://mbhtml.yundasys.com/gateway/ydmb-integral/ydintegral/integralLottery', headers=self.DrawHeaders, json=res_data) print(f"doDraw status_code={resp.status_code}") if resp.status_code==200: res_info = resp.json() if res_info['code'] == 200: msg = res_info['message'] prizeName = res_info['data']['prizeName'] Log(f'>>{msg},获得{prizeName}') else: # Log(f">>{res_info['message']}") Log(f">抽奖失败 参数有问题 或者验证码有问题 {resp.json()} {res_info['message']}") else: Log(f">抽奖失败 404 或token失效 或者验证码有问题") def main(self): Log(f"\n开始执行第{self.index}个账号--------------->>>>>") if not self.get_point(): return False self.sign() self.get_TaskList() self.getDrawInfo() self.sendMsg() return True def sendMsg(self): if self.send_UID: push_res = CHERWIN_TOOLS.wxpusher(self.send_UID, one_msg, APP_NAME) print(push_res) def down_file(filename, file_url): print(f'开始下载:{filename},下载地址:{file_url}') try: response = requests.get(file_url, verify=False, timeout=10) response.raise_for_status() with open(filename + '.tmp', 'wb') as f: f.write(response.content) print(f'【{filename}】下载完成!') # 检查临时文件是否存在 temp_filename = filename + '.tmp' if os.path.exists(temp_filename): # 删除原有文件 if os.path.exists(filename): os.remove(filename) # 重命名临时文件 os.rename(temp_filename, filename) print(f'【{filename}】重命名成功!') return True else: print(f'【{filename}】临时文件不存在!') return False except Exception as e: print(f'【{filename}】下载失败:{str(e)}') return False def import_Tools(): global CHERWIN_TOOLS,ENV, APP_INFO, TIPS, TIPS_HTML, AuthorCode import CHERWIN_TOOLS ENV, APP_INFO, TIPS, TIPS_HTML, AuthorCode = CHERWIN_TOOLS.main(APP_NAME, local_script_name, ENV_NAME,local_version) if __name__ == '__main__': APP_NAME = '韵达快递小程序' ENV_NAME = 'YDKD' CK_NAME = 'Authorization' print(f''' ✨✨✨ {APP_NAME}签到✨✨✨ ✨ 功能: 积分签到 积分抽奖 ✨ 抓包步骤: 打开{APP_NAME} 授权登陆 打开抓包工具 找URl请求头带{CK_NAME} 复制里面的{CK_NAME}参数值 参数示例:oPJUI0eZ5cm8BpR7xxxx ✨ 设置青龙变量: export {ENV_NAME}='{CK_NAME}参数值'多账号#或&分割 export SCRIPT_UPDATE = 'False' 关闭脚本自动更新,默认开启 ✨ ✨ 注意:抓完CK没事儿别打开小程序,重新打开小程序请重新抓包 ✨ 推荐cron:0 6 * * * ✨✨✨ @Author CHERWIN✨✨✨ ''') local_script_name = os.path.basename(__file__) local_version = '2024.05.15' if IS_DEV: import_Tools() else: if os.path.isfile('CHERWIN_TOOLS.py'): import_Tools() else: if down_file('CHERWIN_TOOLS.py', 'https://github.com/CHERWING/CHERWIN_SCRIPTS/raw/main/CHERWIN_TOOLS.py'): print('脚本依赖下载完成请重新运行脚本') import_Tools() else: print( '脚本依赖下载失败,请到https://github.com/CHERWING/CHERWIN_SCRIPTS/raw/main/CHERWIN_TOOLS.py下载最新版本依赖') exit() print(TIPS) token = '' token = ENV if ENV else token if not token: print(f"未填写{ENV_NAME}变量\n青龙可在环境变量设置 {ENV_NAME} 或者在本脚本文件上方将{CK_NAME}填入token =''") exit() tokens = CHERWIN_TOOLS.ENV_SPLIT(token) # print(tokens) if len(tokens) > 0: print(f"\n>>>>>>>>>>共获取到{len(tokens)}个账号<<<<<<<<<<") access_token = [] for index, infos in enumerate(tokens): s = requests.session() s.verify = False run_result = RUN(infos, index).main() if not run_result: continue if send: send(f'{APP_NAME}挂机通知', send_msg + TIPS_HTML)