Showing preview only (329K chars total). Download the full file or copy to clipboard to get everything.
Repository: testerSunshine/12306
Branch: master
Commit: a495af88346a
Files: 79
Total size: 57.7 MB
Directory structure:
gitextract_ltfh64ff/
├── .dockerignore
├── .github/
│ └── ISSUE_TEMPLATE/
│ ├── bug_report.md
│ └── feature_request.md
├── .gitignore
├── 12306.image.model.h5
├── Dockerfile
├── Dockerfile37
├── LICENSE
├── README.md
├── TickerConfig.py
├── UnitTest/
│ ├── TestAll.py
│ └── __init__.py
├── Update.md
├── __init__.py
├── agency/
│ ├── __init__.py
│ ├── agency_tools.py
│ ├── cdn_utils.py
│ └── proxy_list
├── cdn_list
├── config/
│ ├── AutoSynchroTime.py
│ ├── TicketEnmu.py
│ ├── __init__.py
│ ├── configCommon.py
│ ├── emailConf.py
│ ├── getCookie.py
│ ├── logger.py
│ ├── pushbearConf.py
│ ├── serverchanConf.py
│ └── urlConf.py
├── docker-compose.yml
├── docker_install_centos.sh
├── filter_cdn_list
├── init/
│ ├── __init__.py
│ ├── login.py
│ └── select_ticket_info.py
├── inter/
│ ├── AutoSubmitOrderRequest.py
│ ├── ChechFace.py
│ ├── CheckOrderInfo.py
│ ├── CheckRandCodeAnsyn.py
│ ├── CheckUser.py
│ ├── ConfirmHB.py
│ ├── ConfirmSingleForQueue.py
│ ├── ConfirmSingleForQueueAsys.py
│ ├── GetPassCodeNewOrderAndLogin.py
│ ├── GetPassengerDTOs.py
│ ├── GetQueueCount.py
│ ├── GetQueueCountAsync.py
│ ├── GetRandCode.py
│ ├── GetRepeatSubmitToken.py
│ ├── GetSuccessRate.py
│ ├── LiftTicketInit.py
│ ├── LoginAysnSuggest.py
│ ├── LoginConf.py
│ ├── PassengerInitApi.py
│ ├── Query.py
│ ├── QueryOrderWaitTime.py
│ ├── SubmitOrderRequest.py
│ └── __init__.py
├── model.v2.0.h5
├── myException/
│ ├── PassengerUserException.py
│ ├── UserPasswordException.py
│ ├── __init__.py
│ ├── balanceException.py
│ ├── ticketConfigException.py
│ ├── ticketIsExitsException.py
│ └── ticketNumOutException.py
├── myUrllib/
│ ├── MySocketUtils.py
│ ├── __init__.py
│ └── httpUtils.py
├── requirements-docker37.txt
├── requirements.txt
├── run.py
├── station_name.txt
├── tmp/
│ ├── __init__.py
│ └── log/
│ └── __init__.py
└── verify/
├── __init__.py
├── localVerifyCode.py
├── mlearn_for_image.py
└── pretreatment.py
================================================
FILE CONTENTS
================================================
================================================
FILE: .dockerignore
================================================
**/*.html
**/*.pyc
**/*.yaml
**/*.log
**/*~
**/.DS_Store
**/Thumbs.db
*.png
.idea/
.git/
.github/
*.md
UnitTest/
uml/
*.h5
================================================
FILE: .github/ISSUE_TEMPLATE/bug_report.md
================================================
---
name: Bug report
about: Create a report to help us improve
title: ''
labels: ''
assignees: ''
---
**描述问题**
```
A clear and concise description of what the bug is.
```
**重现步骤**
```
步骤一
```
**截图&日志**
**环境信息**
- windows
- python3.7.1
- 订票小助手版本 1.1.101
**额外的备注**
- Add any other context about the problem here.
================================================
FILE: .github/ISSUE_TEMPLATE/feature_request.md
================================================
---
name: Feature request
about: Suggest an idea for this project
title: ''
labels: ''
assignees: ''
---
**Is your feature request related to a problem? Please describe.**
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
**Describe the solution you'd like**
A clear and concise description of what you want to happen.
**Describe alternatives you've considered**
A clear and concise description of any alternative solutions or features you've considered.
**Additional context**
Add any other context or screenshots about the feature request here.
================================================
FILE: .gitignore
================================================
*.html
*.pyc
*.yaml
*.log
.idea/
tkcode.png
================================================
FILE: 12306.image.model.h5
================================================
[File too large to display: 57.4 MB]
================================================
FILE: Dockerfile
================================================
FROM python:2.7.15
WORKDIR /usr/src/app
ADD . /usr/src/app
ENV DEBIAN_FRONTEND noninteractive
ENV TZ Asia/Shanghai
## install python requirements
RUN pip install -i https://pypi.tuna.tsinghua.edu.cn/simple pyspider --no-cache-dir -r requirements.txt
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone
## install ntpdate, not accept but saving code
#RUN echo 'deb http://mirrors.163.com/debian/ jessie main non-free contrib \
# deb http://mirrors.163.com/debian/ jessie-updates main non-free contrib \
# deb http://mirrors.163.com/debian-security/ jessie/updates main non-free contrib' > /etc/apt/sources.list \
# && apt-get update\
# && apt-get install ntpdate -y \
#EXPOSE 5010
CMD [ "python", "run.py" ]
#ENTRYPOINT [ "python", "run.py" ]
================================================
FILE: Dockerfile37
================================================
FROM python:3.7-slim-buster
ARG CDV=77.0.3865.40
RUN sed -i 's/deb.debian.org/ftp.cn.debian.org/g' /etc/apt/sources.list
RUN apt-get -y update && apt-get install -y \
fonts-liberation \
libappindicator3-1 \
libasound2 \
libatk-bridge2.0-0 \
libatk1.0-0 \
libatspi2.0-0 \
libcups2 \
libdbus-1-3 \
libgtk-3-0 \
libnspr4 \
libnss3 \
libx11-xcb1 \
libxcomposite1 \
libxcursor1 \
libxdamage1 \
libxfixes3 \
libxi6 \
libxrandr2 \
libxss1 \
libxtst6 \
lsb-release \
unzip \
wget \
xdg-utils \
&& apt-get clean && rm -rf /var/lib/apt/lists/*
ENV TZ Asia/Shanghai
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone
WORKDIR /usr/src/app
RUN wget -q https://dl.lancdn.com/landian/soft/chrome/m/77.0.3865.120_amd64.deb && \
dpkg -i 77.0.3865.120_amd64.deb && rm -f 77.0.3865.120_amd64.deb
RUN wget -q https://npm.taobao.org/mirrors/chromedriver/$CDV/chromedriver_linux64.zip && \
unzip chromedriver_linux64.zip && rm -f chromedriver_linux64.zip
## install python requirements
COPY requirements-docker37.txt ./
RUN pip install -i https://pypi.tuna.tsinghua.edu.cn/simple --no-cache-dir -r requirements-docker37.txt
COPY . .
CMD [ "sh", "-c", "python run.py c && python run.py r" ]
================================================
FILE: LICENSE
================================================
MIT License
Copyright (c) 2017
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
================================================
FILE: README.md
================================================
### 12306 购票小助手
#### python版本
- [ ] 2.7.10 - 2.7.15
- [x] 3.6 - 3.7.4
- [ ] 2.7.9
#### 已有功能
- [x] 自动打码
- [x] 自动登录
- [x] 准点预售和捡漏
- [x] 智能候补
- [x] 邮件通知
- [x] server酱通知
#### 依赖库
- 验证码目前可以本地识别,需要下载模型,放于项目根目录,全部代码来源于此项目 [传送门](https://github.com/zhaipro/easy12306),表示感谢
```
1. 模型下载链接:https://pan.baidu.com/s/1rS155VjweWVWIJogakechA 密码:bmlm
群里面也可以下载
2. git仓库下载:https://github.com/testerSunshine/12306model.git
```
- 自托管云打码服务器搭建:[12306_code_server](https://github.com/YinAoXiong/12306_code_server)
- 如果大家有空闲的服务器,可搭建之后在这个 [issues](https://github.com/testerSunshine/12306/issues/446) 里面填入自己的服务器(请注意服务器安全!)
- 项目依赖 [requirements.txt](requirements.txt)
- 安装方法x:
- root用户(避免多python环境产生问题): `pip3 install -i https://pypi.tuna.tsinghua.edu.cn/simple -r requirements.txt`
- 非root用户(避免安装和运行时使用了不同环境): `pip3 install -i https://pypi.tuna.tsinghua.edu.cn/simple -r requirements.txt`
- 许多windows的用户装不了tensorflow的话,可以适当降低版本或者升高版本都是可以的
```
1. tensorflow的兼容版本 1.14.0rc\1.14.0rc\1.15.0\1.15.0rc
以上版本都测试无问题
2. 如果pip代理的清华源无法下载,可以更换其他源解决此问题
```
#### 项目使用说明
- 服务器启动:
- 修改[配置](TickerConfig.py)文件
- 可以配置邮箱,配置邮箱的格式在[配置](TickerConfig.py)里面可以看到ex
```
# 测试邮箱和server酱是否可用, server酱测试的前提是server酱开关开启
# 可以配置server酱提醒(推荐)[配置教程](https://www.jianshu.com/p/8d10b5b9c4e3)
# 用python3 还是python 完全取决于安装的时候配置的环境变量是否为python3,以下启动默认环境变量为python3
python3 run.py t
```
- 配置[配置](TickerConfig.py)文件的时候,需注意空格和遵循python语法格式
- 启动前请先筛选cdn,这点很`重要`
```
python3 run.py c
```
- 启动服务
```
python3 run.py r
```
- 如果你不知道如何操作,下面的命令可能会帮助你
```
python3 run.py -h
——————————————————————————
sage: run.py [-h] operate
positional arguments:
operate r: 运行抢票程序, c: 过滤cdn, t: 测试邮箱和server酱,server酱
```
- 如果你的服务器安装了docker与docker-compose, 那么你可以忽略上面的**所有**步骤,直接按以下步骤操作,即可开始抢票:
- 前提条件:
- 请确认你安装的docker版本为18.09及以上: `docker -v`
- 请确认你安装的docker-compose版本为1.23.2及以上: `docker-compose -v`
- 请根据自己需要修改好配置文件:`TickerConfig.py`
- 请修改配置文件`TickerConfig.py`中的变量`AUTO_CODE_TYPE`和`HOST`,`AUTO_CODE_TYPE`改为`3`, HOST改为`"captcha:80"`(这里很重要,这是本地打码服务器的配置)
- 运行命令:
- 开始抢票:`docker-compose up --build -d`
- 停止抢票:`docker-compose down`
- 查看抢票log: `docker logs --follow ticket`
#### 目录对应说明
- agency - cdn代理
- config - 项目配置
- verify - 自动打码
- init - 项目主运行目录
- inter - 接口
- myException - 异常
- myUrllib request网络请求库
#### 思路图
- 
#### 项目声明:
- 本软件只供学习交流使用,勿作为商业用途,交流群号
- 1群:286271084(已满)
- 2群:649992274(已满)
- 3群:632501142(已满)
- 4群: 606340519(已满)
- 5群: 948526733(已满)
- 7群: 660689659(已满)
- 8群: 620629239(已满)
- 6群: 608792930(未满)
- 9群: 693035807(未满)
- 请不要重复加群,一个群就可以了,把机会留给更多人
- **进群先看公告!!!进群先看公告!!!进群先看公告!!! 重要的事情说三遍**
- 能为你抢到一张回家的票,是我最大的心愿
#### 日志列子
- 成功log,如果是购票失败的,请带上失败的log给我,我尽力帮你调,也可加群一起交流,程序只是加速买票的过程,并不一定能买到票
```
正在第355次查询 乘车日期: 2018-02-12 车次G4741,G2365,G1371,G1377,G1329 查询无票 代理设置 无 总耗时429ms
车次: G4741 始发车站: 上海 终点站: 邵阳 二等座:有
正在尝试提交订票...
尝试提交订单...
出票成功
排队成功, 当前余票还剩余: 359 张
正在使用自动识别验证码功能
验证码通过,正在提交订单
提交订单成功!
排队等待时间预计还剩 -12 ms
排队等待时间预计还剩 -6 ms
排队等待时间预计还剩 -7 ms
排队等待时间预计还剩 -4 ms
排队等待时间预计还剩 -4 ms
恭喜您订票成功,订单号为:EB52743573, 请立即打开浏览器登录12306,访问‘未完成订单’,在30分钟内完成支付!
```
#### 使用帮助(一些安装问题和使用反馈较多的问题):
- 测试邮箱是否可用 [邮箱配置问题看issues](https://github.com/testerSunshine/12306/issues/107)
- 学生票issues [学生票修改](https://github.com/testerSunshine/12306/issues/47)
- 依赖安装不对的问题(ImportError)[requirements.txt问题](https://github.com/testerSunshine/12306/issues/91)
- 若快豆子疑问 [点我](https://github.com/testerSunshine/12306/issues/67)
- IOError: 【Errno 0】 Error 问题 [点我](https://github.com/testerSunshine/12306/issues/159)
- 测试下单接口是否可用,有两个下单接口,随便用哪个都ok
- 如果下载验证码过期或者下载失败的问题,应该是12306封ip的策略,多重试几次,12306现在封服务器(阿里云和腾讯云)ip比较严重,尽量不要放在服务器里面
- 目前12306对服务器ip比较敏感,大家还是在自己家里挂着吧
- 自动更换ip软件目前已支持TPLINK和小米路由器,只限家庭网络[点我跳转](https://github.com/testerSunshine/AutoRouterIP)
#### 感谢一下小伙伴对本项目提供的帮助
- @sun7127@126.com
- @ 才
- @[MonsterTan](https://github.com/MonsterTan)
- 以及所有为此项目提供pr的同学
#### 更新日志
- [更新日志](Update.md)
================================================
FILE: TickerConfig.py
================================================
# -*- coding=utf-8 -*-
# 关于软件使用配置说明,一定要看!!!
# ps: 如果是候补车票,需要通过人证一致性核验的用户及激活的“铁路畅行”会员可以提交候补需求,请您按照操作说明在铁路12306app.上完成人证核验
# 关于候补了之后是否还能继续捡漏的问题在此说明: 软件为全自动候补加捡漏,如果软件候补成功则会停止抢票,发出邮件通知,但是不会影响你继续捡漏,
# 如果这个时候捡漏捡到的话,也是可以付款成功的,也就是说,捡漏+候补,可以最大程度提升抢票成功率
# 刷票模式:1=刷票 2=候补+刷票
TICKET_TYPE = 1
# 出发日期(list) "2018-01-06", "2018-01-07"
STATION_DATES = [
"2020-01-18"
]
# 填入需要购买的车次(list),"G1353"
# 修改车次填入规则,注:(以前设置的车次逻辑不变),如果车次填入为空,那么就是当日乘车所有车次都纳入筛选返回
# 不填车次是整个list为空才算,如果不是为空,依然会判断车次的,这种是错误的写法 [""], 正确的写法 []
STATION_TRAINS = []
# 出发城市,比如深圳北,就填深圳就搜得到
FROM_STATION = "广州南"
# 到达城市 比如深圳北,就填深圳就搜得到
TO_STATION = "隆回"
# 座位(list) 多个座位ex:
# "商务座",
# "一等座",
# "二等座",
# "特等座",
# "软卧",
# "硬卧",
# "硬座",
# "无座",
# "动卧",
SET_TYPE = ["二等座"]
# 当余票小于乘车人,如果选择优先提交,则删减联系人和余票数一致在提交
# bool
IS_MORE_TICKET = True
# 乘车人(list) 多个乘车人ex:
# "张三",
# "李四"
TICKET_PEOPLES = []
# 12306登录账号
USER = ""
PWD = ""
# 加入小黑屋时间默认为5分钟,此功能为了防止僵尸票导致一直下单不成功错过正常的票
TICKET_BLACK_LIST_TIME = 5
# 自动打码
IS_AUTO_CODE = True
# 设置2本地自动打码,需要配置tensorflow和keras库,3为云打码,由于云打码服务器资源有限(为2h4C的cpu服务器),请不要恶意请求,不然只能关闭服务器
# ps: 请不要一直依赖云服务器资源,在此向所有提供服务器同学表示感谢
AUTO_CODE_TYPE = 3
# 此处设置云打码服务器地址,如果有自建的服务器,可以自行更改
HOST = "120.77.154.140:8000"
REQ_URL = "/verify/base64/"
HTTP_TYPE = "http"
# HOST="12306.yinaoxiong.cn" #备用服务器稳定性较差
# REQ_URL="/verify/base64/"
# HTTP_TYPE="https"
# 邮箱配置,如果抢票成功,将通过邮件配置通知给您
# 列举163
# email: "xxx@163.com"
# notice_email_list: "123@qq.com"
# username: "xxxxx"
# password: "xxxxx
# host: "smtp.163.com"
# 列举qq ,qq设置比较复杂,需要在邮箱-->账户-->开启smtp服务,取得授权码==邮箱登录密码
# email: "xxx@qq.com"
# notice_email_list: "123@qq.com"
# username: "xxxxx"
# password: "授权码"
# host: "smtp.qq.com"
EMAIL_CONF = {
"IS_MAIL": True,
"email": "",
"notice_email_list": "",
"username": "",
"password": "",
"host": "smtp.qq.com",
}
# 是否开启 server酱 微信提醒, 使用前需要前往 http://sc.ftqq.com/3.version 扫码绑定获取 SECRET 并关注获得抢票结果通知的公众号
SERVER_CHAN_CONF = {
"is_server_chan": False,
"secret": ""
}
# 是否开启cdn查询,可以更快的检测票票 1为开启,2为关闭
IS_CDN = 1
# 下单接口分为两种,1 模拟网页自动捡漏下单(不稳定),2 模拟车次后面的购票按钮下单(稳如老狗)
ORDER_TYPE = 2
# 下单模式 1 为预售,整点刷新,刷新间隔0.1-0.5S, 然后会校验时间,比如12点的预售,那脚本就会在12.00整检票,刷新订单
# 2 是捡漏,捡漏的刷新间隔时间为0.5-3秒,时间间隔长,不容易封ip
ORDER_MODEL = 1
# 是否开启代理, 0代表关闭, 1表示开始
# 开启此功能的时候请确保代理ip是否可用,在测试放里面经过充分的测试,再开启此功能,不然可能会耽误你购票的宝贵时间
# 使用方法:
# 1、在agency/proxy_list列表下填入代理ip
# 2、测试UnitTest/TestAll/testProxy 测试代理是否可以用
# 3、开启代理ip
IS_PROXY = 0
# 预售放票时间, 如果是捡漏模式,可以忽略此操作
OPEN_TIME = "12:59:57"
# 1=使用selenium获取devicesID
# 2=使用网页端/otn/HttpZF/logdevice获取devicesId,这个接口的算法目前可能有点问题,如果登录一直302的请改为配置1
# 3=自己打开浏览器在headers-Cookies中抓取RAIL_DEVICEID和RAIL_EXPIRATION,这个就不用配置selenium
COOKIE_TYPE = 3
# 如果COOKIE_TYPE=1,则需配置chromeDriver路径,下载地址http://chromedriver.storage.googleapis.com/index.html
# chromedriver配置版本只要和chrome的大版本匹配就行
CHROME_PATH = "/usr/src/app/chromedriver"
# 为了docker37 准备的环境变量,windows环境可以不用管这个参数
CHROME_CHROME_PATH = "/opt/google/chrome/google-chrome"
# 如果COOKIE_TYPE=3, 则需配置RAIL_EXPIRATION、RAIL_DEVICEID的值
RAIL_EXPIRATION = ""
RAIL_DEVICEID = ""
# RAIL_EXPIRATION = "1577034103293"
# RAIL_DEVICEID = "CDno29Erc_Pf3FSXb4dzq-Op64EhWrsi5yUZKVIKR1MAfYo2qFlCeXD8VkexY7_1qg-ClV-fE8j9jgVlPZxRh3wVc2iqLe_5A8sdr62qZx4B22JPF8lFCjpgTKZ5ODW90HJd5tiQsJ1KR9nOqHRxHj1FT5LEIwfw"
# 1=>为一直随机ua,2->只启动的时候随机一次ua
RANDOM_AGENT = 2
PASSENGER_TICKER_STR = {
'一等座': 'M',
'特等座': 'P',
'二等座': 'O',
'商务座': 9,
'硬座': 1,
'无座': 1,
'软座': 2,
'软卧': 4,
'硬卧': 3,
}
# 保护12306官网请求频率,设置随机请求时间,原则为5分钟不大于80次
# 最大间隔请求时间
MAX_TIME = 3
# 最小间隔请求时间
MIN_TIME = 1
# 软件版本
RE_VERSION = "1.2.004"
================================================
FILE: UnitTest/TestAll.py
================================================
# coding=utf-8
import base64
import threading
import unittest
from collections import OrderedDict
import requests
import TickerConfig
from agency.agency_tools import proxy
from config.emailConf import sendEmail
from config.serverchanConf import sendServerChan
from inter.LiftTicketInit import liftTicketInit
def _set_header_default():
header_dict = OrderedDict()
header_dict["Accept"] = "*/*"
header_dict["Accept-Encoding"] = "gzip, deflate"
header_dict["X-Requested-With"] = "superagent"
header_dict[
"User-Agent"] = "Mozilla/5.0 (iPhone; CPU iPhone OS 11_0 like Mac OS X) AppleWebKit/604.1.38 (KHTML, like Gecko) Version/11.0 Mobile/15A372 Safari/604.1"
header_dict["Content-Type"] = "application/x-www-form-urlencoded; charset=UTF-8"
class testAll(unittest.TestCase):
def testProxy(self):
"""
测试代理是否可用
:return:
"""
_proxy = proxy()
proxie = _proxy.setProxy()
url = "http://httpbin.org/ip"
rsp = requests.get(url, proxies=proxie, timeout=5, headers=_set_header_default()).content
print(u"当前代理ip地址为: {}".format(rsp))
def testEmail(self):
"""
实测邮箱是否可用
:return:
"""
sendEmail(u"订票小助手测试一下")
# def testConfig(self):
# """
# 测试config是否配置正确
# :return:
# """
def testServerChan(self):
"""
实测server酱是否可用
:return:
"""
sendServerChan(u"server酱 微信通知测试一下")
def testUserAgent(self):
"""
测试UserAgent
:return:
"""
from fake_useragent import UserAgent
for i in range(10000):
ua = UserAgent(verify_ssl=False)
print(ua.random)
def testVerfyImage(self):
"""
测试模型加载识别
:return:
"""
from verify.localVerifyCode import Verify
v = Verify()
with open('../tkcode.png', 'rb') as f:
base64Image = base64.b64encode(f.read())
for i in range(5):
t = threading.Thread(target=v.verify, args=(base64Image,))
t.start()
def testRemoteVerfy(self):
"""
测试打码是否可用
:return:
"""
import requests
import time
while True:
try:
starttime = time.time()
rsp = requests.post(url="http://120.77.154.140:8000/verify/base64/",
data={
'imageFile': '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'},
timeout=60,
)
print(rsp.content)
print(f"响应时间{time.time()-starttime}m")
except:
pass
def testCdn(self):
"""
测试cdn筛选
:return:
"""
cdn = ["60.9.0.19", "60.9.0.20", "113.16.212.251", "36.250.248.27"]
from inter.LiftTicketInit import liftTicketInit
from init.select_ticket_info import select
from config.getCookie import getDrvicesID
s = select()
s.httpClint.cdn = cdn[3]
getDrvicesID(s)
liftTicketInit(s).reqLiftTicketInit()
if __name__ == '__main__':
unittest.main()
================================================
FILE: UnitTest/__init__.py
================================================
================================================
FILE: Update.md
================================================
- 2017.5.13跟新
- 增加登陆错误判断(密码错误&ip校验)
- 修改queryOrderWaitTime,校验orderId字段bug,校验msg字段bug,校验messagesbug
- 修改checkQueueOrder 校验 data 字段的列表推导式bug
- 增加代理ip方法,目前已可以过滤有用ip
- 2018.1.7 号更新
- 增加自动配置
```
#station_date:出发日期,格式ex:2018-01-06
#from_station: 始发站
#to_station: 到达站
#set_type: 坐席(商务座,二等座,特等座,软卧,硬卧,硬座,无座)
#is_more_ticket:余票不足是否自动提交
#select_refresh_interval:刷新间隔时间,1为一秒,0.1为100毫秒,以此类推
#ticke_peoples: 乘客
#damatu:打码图账号,用于自动登录
```
- 优化订票流程
- 支持自动刷票,自动订票
- 2018.1.8 更新
- 增加小黑屋功能
- 修复bug若干
- 增加多账号同时订票功能
- 增加按照选定车次筛选购买车次
- 2018.1.9 更新
- 增加手动打码,只是登录接口,完全不用担心提交票的效率问题,要挂linux系统的话,还是去注册个打码兔吧
```
思路
1.调用PIL显示图片
2.图片位置说明,验证码图片中每个图片代表一个下标,依次类推,1,2,3,4,5,6,7,8
3.控制台输入对应下标,按照英文逗号分开,即可手动完成打码,
```
- 修改无座和硬座的座位号提交是个字符串的问题
- 增加校验下单需要验证码功能
- 增强下单成功判断接口校验
- 2018.1.10 更新
- 优化查票流程
- 修改二等座的余票数返回为字符串的问题
- 优化订单查询bug
- 2018.1.12更新
- 优化抢票页面逻辑
-增强代码稳定性
- 2018.1.13更新
- 修改下单验证码功能
- 优化大量调用user接口导致联系人不能用,理论加快订票速度
- 增加邮箱功能
```
#is_email: 是否需要邮件通知 ex: True or False 切记,邮箱加完一定要到config目录下测试emailConf功能是否正常
#email: 发送的邮箱地址 ex: 1@qq.com
#notice_email_list: 被通知人邮箱 ex: 2@qq.com
#username: 邮箱账号
#password: 邮箱密码
#host: 邮箱地址
```
- 2018.1.14更新
- 优化订票流程
- 优化挂机功能
- 修改之前程序到11点自动跳出功能,现在修改为到早上7点自动开启刷票
- 需要开启打码兔代码功能,is_auto_code 设置为True
- 增加异常判断
- 2018.1.15更新
- 增加捡漏自动检测是否登录功能,建议捡漏不要刷新太快,2S最好,否则会封IP
- 优化提交订单有很大记录无限排队的情况,感谢群里的小伙伴提供的思路
- 修改休眠时间为早上6点
- 2018.1.20更新,好久没跟新了,群里的小伙伴说登录不行了,今晚抽空改了一版登录,妥妥的
- 更新新版登录功能,经测试,更稳定有高效
- 优化手动打码功能
- 更新请求第三方库
- 优化若干代码,小伙伴尽情的放肆起来
- 2018.1.21跟新
- 修复若干bug
- 合并dev
- 恢复之前因为12306改版引起的订票功能
- 增加派对失败自动取消订单功能
- 优化接口请求规范
- 增加多日期查询,请严格按照yaml格式添加 即可
- 注意:如果多日期查询的话,可能查询时间会比较长
- 增加如果排队时间超过一分钟,自动取消订单
- 2018.1.23更新
- 增加若快平台打码,yaml新增字段auto_code_type,1=打码兔,2=若快 若快注册地址:http://www.ruokuai.com/client/index?6726
- 修改is_auto_code字段为全部是否自动打码字段,也就是说字段为True,则全部自动打码,为False全部手动打码,包括提交订单,注意centOs不可设置手动打码
- 修复bug
- 优化抢票功能
- 2018.1.25更新
- 删除 expect_refresh_interval 无用字段,优化代码
- 2018.1.29更新
- 增加cdn轮训功能,优势, is_cdn=1为开启cdn,2为普通查询
- 能够一定程度躲避封ip
- 查询余票快人一步
- 提交订单快人一步
- 僵尸票可能会少很多
- 兼容Windows cmd命令乱码问题
- 规范12306接口提交参数
- 修改已知bug
- 最后感谢群里提供测试和代码的小伙伴,能为你们买到一张回家的票真的感到灰常开心
- 2018.2.28更新,收12306风控影响,代码可能有时候会进入慢排队,请自行调整刷新参数
- 修改已知bug
- 优化接口提交参数规范
- 2018.8.29更新,解决慢排队问题,增加双订票接口
- 修改已知bug
- 优化代码结构
- 新增第三方库wrapcache,需要重新安装requirements.txt
- 2018.9.21更新,修复已知问题,增加余票不足优先提交功能
- 修改已知bug
- 优化查询攻略
- 优化随机查询1-3秒,经测试很稳定,不会封ip
- 增加余票不足优先提交功能(当余票小于乘车人,如果选择优先提交,则删减联系人和余票数一致在提交)
- 开关为ticket_config.yaml配置文件中is_more_ticket参数
- 2018.12.26更新
- 优化已知bug
- 开启cdn查询
- 自动识别查询接口
- 2019.01.09更新
- ticket_config 配置文件增加order_model字段(下单模式)
- mac和linux服务器自动对点
- 增加预售踩点查询下单,经测试,误差在0.004s
- 2019.01.12更新
- 增加对python3语法的支持
- 删除校验时间很多机器不兼容的问题(win10会阻拦对时功能,导致大面积报错),如果是预售,为了不耽误宝贵的时间,请手动对时
- 2019.01.15更新
- 删除敏感信息打印
- 增加server酱推送购票成功通知
- 修改11点都登录消耗快豆问题
- 增加gui界面
- 2019.04.01更新
- 增加登录接口,解决无法03.31登录
- 2019.04.23更新
- 修复登录302问题,如果下次没来得及更新,请自行抓包修改urlConf中getDevicesId接口即可登录
- 2019.04.23更新
- 更新本地识别
- 2019.08.31更新
- 删除若快打码
- 修复不能下单问题
- 放弃支持python2.7,只支持3.6以上版本
- 2019.09.01更新
- 去除yaml配置文件,改为py文件配置
- 增加候补订单功能
- 新增TICKET_TYPE字段,1=刷票 2=候补
- 目前候补只支持单车次,多乘车人候补,由于目前不是很懂候补的需求,所以暂时这样做
- 2019.09.02更新
- 剔除新增TICKET_TYPE字段,发现此字段冗余,目前改为全天自动候补+捡漏
- 增加候补订单功能
- 优先级:下单再候补
- 2019.09.03更新
- 恢复TICKET_TYPE字段,1=刷票 2=候补+刷票
- 优化候补逻辑
- 候补订单只能在规定车次内候补
- 2019.09.07更新
- 优化候补逻辑
- 去除敏感信息打印
- 2019.09.09更新
- 优化候补逻辑
- 2019.09.15更新
- 增长随机停留时长
- 增长用户心跳时间,减少对服务器压力
- 优化下单逻辑
- 2019.09.18更新
- 修改下单问题
- 优化车次打印
================================================
FILE: __init__.py
================================================
================================================
FILE: agency/__init__.py
================================================
================================================
FILE: agency/agency_tools.py
================================================
# encoding=utf8
import os
import random
import socket
import time
import requests
from bs4 import BeautifulSoup
class proxy:
def __init__(self):
self.proxy_list = []
self.proxy_filter_list = []
def get_proxy(self):
"""
获取未加工代理列表
:return:
"""
User_Agent = 'Mozilla/5.0 (Windows NT 6.3; WOW64; rv:43.0) Gecko/20100101 Firefox/43.0'
header = dict()
header['User-Agent'] = User_Agent
for i in range(1, 5):
time.sleep(1)
url = 'http://www.xicidaili.com/nn/' + str(i)
res = requests.get(url=url, headers=header).content
soup = BeautifulSoup(res, "html.parser")
ips = soup.findAll('tr')
for x in range(1, len(ips)):
ip = ips[x]
tds = ip.findAll("td")
ip_temp = tds[1].contents[0] + ":" + tds[2].contents[0]
print(ip_temp)
self.proxy_list.append(ip_temp)
def filter_proxy(self):
"""
将不可用IP剔除
:return:
"""
socket.setdefaulttimeout(1)
path = os.path.join(os.path.dirname(__file__), './proxy_list')
f = open(path, "w")
head = {
'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/50.0.2661.102 Safari/537.36',
'Connection': 'keep-alive'}
url = "http://icanhazip.com"
proxy_num = 0
for proxy in self.proxy_list:
proxy_temp = {"https": "https://{}".format(proxy)}
try:
req = requests.get(url, proxies=proxy_temp, timeout=2, headers=head).content
print(req)
write_proxy = proxy + "\n"
f.write(write_proxy)
proxy_num += 1
except Exception:
print ("代理链接超时,去除此IP:{0}".format(proxy))
continue
print("总共可使用ip量为{}个".format(proxy_num))
def get_filter_proxy(self):
"""
读取该可用ip文件
:return: 可用ip文件list
"""
path = os.path.join(os.path.dirname(__file__), './proxy_list')
try:
with open(path, "r", encoding="utf-8") as f:
lins = f.readlines()
for i in lins:
p = i.strip("\n")
self.proxy_filter_list.append(p)
except Exception:
with open(path, "r", ) as f:
lins = f.readlines()
for i in lins:
p = i.strip("\n")
self.proxy_filter_list.append(p)
return self.proxy_filter_list
def main(self):
# self.get_proxy()
self.filter_proxy()
def setProxy(self):
"""
开启此功能的时候请确保代理ip是否可用
查询的时候设置代理ip,ip设置格式是ip地址+端口,推荐可用的ip代理池:https://github.com/jhao104/proxy_pool
:return:
"""
ip = self.get_filter_proxy()
setIp = ip[random.randint(0, len(ip) - 1)]
proxie = {
'http': 'http://{}'.format(setIp),
'https': 'http://{}'.format(setIp),
}
return proxie
if __name__ == "__main__":
a = proxy()
print(a.get_filter_proxy())
================================================
FILE: agency/cdn_utils.py
================================================
# encoding=utf8
import datetime
import operator
import os
import requests
from config import urlConf
import threading
from config.urlConf import urls
from myUrllib.httpUtils import HTTPClient
cdn_list = []
class CDNProxy(threading.Thread):
def __init__(self, cdns):
super().__init__()
self.cdns = cdns
self.urlConf = urlConf.urls
self.httpClint = requests
self.city_list = []
self.timeout = 5
def run(self):
for cdn in self.cdns:
http = HTTPClient(0)
url = urls["loginInitCdn"]
http._cdn = cdn.replace("\n", "")
start_time = datetime.datetime.now()
rep = http.send(url)
retTime = (datetime.datetime.now() - start_time).microseconds / 1000
if rep and "message" not in rep and retTime < 3000:
if cdn.replace("\n", "") not in cdn_list: # 如果有重复的cdn,则放弃加入
print(f"加入cdn: {cdn}")
cdn_list.append({"ip": cdn.replace("\n", ""), "time": retTime})
def open_cdn_file(cdnFile):
cdn = []
path = os.path.join(os.path.dirname(__file__), f'../{cdnFile}')
try:
with open(path, "r", encoding="utf-8") as f:
for i in f.readlines():
if i and "kyfw.12306.cn:443" not in i:
cdn.append(i.replace("\n", ""))
return cdn
except Exception:
with open(path, "r") as f:
for i in f.readlines():
if i and "kyfw.12306.cn:443" not in i:
cdn.append(i.replace("\n", ""))
return cdn
def sortCdn():
"""
对cdn进行排序
:return:
"""
ips = []
cs = sorted(cdn_list, key=operator.itemgetter('time'))
for c in cs:
print(f"当前ip: {c['ip']}, 延时: {c['time']}")
ips.append(c["ip"])
return ips
def filterCdn():
"""
过滤cdn, 过滤逻辑为当前cdn响应值小于1000毫秒
过滤日志:
加入cdn: 116.77.75.146
:return:
"""
cdns = open_cdn_file("cdn_list")
cdnss = [cdns[i:i + 50] for i in range(0, len(cdns), 50)]
cdnThread = []
for cdn in cdnss:
t = CDNProxy(cdn)
cdnThread.append(t)
for cdn_t in cdnThread:
cdn_t.start()
for cdn_j in cdnThread:
cdn_j.join()
print(f"当前有效cdn个数为: {len(cdn_list)}")
if cdn_list:
ips = sortCdn()
path = os.path.join(os.path.dirname(__file__), f'../filter_cdn_list')
f = open(path, "a+")
f.seek(0)
f.truncate()
f.writelines("")
for ip in ips:
f.writelines(f"{ip}\n")
f.close()
if __name__ == '__main__':
filterCdn()
================================================
FILE: agency/proxy_list
================================================
119.101.114.196:9999
================================================
FILE: cdn_list
================================================
112.123.33.18
112.28.196.75
112.28.196.100
112.29.227.103
112.29.227.250
112.29.227.106
112.30.197.137
112.28.196.54
112.30.197.16
112.28.196.251
112.28.196.249
112.28.196.53
112.28.196.74
112.30.198.110
112.29.227.251
112.29.227.107
112.30.197.250
112.30.197.17
60.174.243.155
211.162.1.168
60.174.243.190
112.29.227.45
60.174.243.156
60.174.243.157
112.29.227.102
112.28.196.101
103.254.191.203
103.254.189.230
61.132.238.93
61.132.238.126
61.132.238.94
60.174.243.166
111.206.186.37
114.112.160.123
61.132.238.91
103.254.191.210
106.120.178.253
111.200.194.219
106.120.178.20
114.112.172.52
103.254.189.229
106.120.178.22
114.112.160.201
114.112.160.29
114.112.172.123
114.112.160.31
114.112.172.56
114.112.172.55
114.112.160.30
61.132.238.92
114.112.160.124
114.112.172.241
114.112.160.253
122.70.142.175
114.112.172.54
114.112.172.53
114.113.88.86
122.70.142.176
114.113.88.85
106.120.178.19
114.112.172.58
114.113.88.84
114.113.88.88
122.70.142.148
122.70.142.147
114.112.172.57
114.113.88.123
114.112.172.60
114.112.172.61
114.112.172.59
114.112.160.32
159.226.225.149
114.113.88.89
159.226.225.154
36.110.141.253
39.134.134.74
159.226.225.139
159.226.225.140
114.113.88.87
122.70.142.252
61.149.22.254
39.134.134.77
61.149.22.35
61.149.22.34
61.149.9.150
61.149.9.170
61.149.9.239
112.47.14.198
112.47.14.199
112.47.20.250
112.47.14.201
112.47.20.67
112.47.20.77
112.47.14.200
112.47.20.88
112.47.20.68
112.47.27.132
112.47.20.89
112.47.20.209
112.47.20.79
112.47.27.131
112.47.56.174
112.47.27.172
112.47.56.117
112.51.121.107
112.47.56.118
112.51.125.243
112.5.62.12
112.51.121.108
112.51.125.254
112.51.125.33
112.51.125.53
112.51.125.251
112.51.125.54
112.51.125.62
112.51.125.252
112.51.121.251
112.51.125.65
117.27.241.110
112.51.125.74
117.27.241.112
112.51.125.73
117.27.241.125
117.27.241.113
117.27.241.126
117.27.241.111
117.27.241.213
117.27.241.214
117.27.241.254
117.27.241.218
117.27.241.76
117.27.245.223
117.27.245.225
117.27.245.227
117.27.245.254
117.27.245.52
117.27.245.97
117.27.245.54
125.77.130.247
125.77.130.251
125.77.130.44
125.77.130.45
125.77.130.46
125.77.130.47
125.77.130.48
125.77.130.49
125.77.140.120
125.77.140.121
125.77.140.30
125.77.140.42
125.77.140.53
125.77.140.54
125.77.140.55
125.77.140.64
125.77.140.91
125.77.140.92
125.77.140.95
125.77.147.254
125.77.147.62
125.77.147.67
125.77.147.63
125.77.147.66
125.77.147.68
125.77.147.69
125.77.147.83
125.77.147.80
125.77.147.82
125.77.147.88
175.43.20.64
175.43.20.118
183.253.58.102
175.43.20.65
183.253.58.137
183.253.58.103
183.253.58.253
183.253.58.76
27.148.151.175
27.148.151.174
27.148.151.177
27.148.151.176
27.148.151.178
27.148.151.179
27.148.151.180
27.148.151.251
27.155.72.251
27.148.151.72
27.155.72.44
27.155.72.47
36.250.233.185
36.250.233.208
36.250.233.209
36.250.233.210
36.250.233.211
36.250.233.212
36.250.233.214
36.250.233.228
36.250.248.218
36.250.248.217
36.250.233.254
36.250.248.210
36.250.248.219
36.250.248.220
36.250.248.221
36.250.248.223
36.250.248.222
36.250.248.25
36.250.248.24
36.250.248.252
36.250.248.254
36.250.248.27
36.250.248.55
36.250.248.56
36.250.74.128
36.250.74.127
36.250.74.244
59.56.30.250
59.56.30.51
118.180.15.252
59.56.30.52
118.180.15.33
118.180.15.86
118.180.15.87
118.180.57.100
118.180.57.101
118.180.57.250
125.74.58.134
125.74.58.135
125.74.58.136
125.74.58.137
125.74.58.138
125.74.58.254
125.75.35.173
125.75.35.189
125.75.35.190
180.95.178.250
202.201.14.181
180.95.178.38
112.90.133.246
112.90.133.247
112.90.133.253
112.90.135.228
112.90.135.229
112.90.135.238
112.90.135.240
112.90.135.244
112.90.135.91
112.90.135.92
112.90.135.93
112.90.135.94
112.90.135.95
112.90.135.96
112.90.135.97
112.90.135.98
112.90.135.99
113.104.14.227
113.104.14.23
113.104.14.24
113.104.14.248
113.104.14.25
113.104.14.26
113.104.14.27
113.104.14.28
222.186.145.51
122.228.237.248
222.186.145.54
222.44.151.24
150.138.167.51
58.216.21.63
222.186.145.52
121.46.247.75
60.210.23.26
221.235.187.119
59.63.221.41
60.210.23.116
61.147.226.46
222.186.145.53
61.147.226.48
122.224.186.223
60.210.23.29
221.235.187.67
58.221.78.58
183.146.22.142
183.146.22.139
221.235.187.121
121.46.247.72
122.225.83.26
222.186.145.251
153.101.208.89
61.147.227.53
113.104.14.29
113.104.14.32
113.104.14.30
113.104.14.31
113.104.14.33
113.104.14.34
116.199.127.50
116.199.127.54
116.77.73.164
116.77.73.165
116.199.127.55
116.77.75.133
116.77.75.137
116.199.127.56
116.77.75.138
116.77.75.145
116.77.75.144
116.77.75.169
116.77.75.146
116.77.75.147
116.77.75.170
120.198.197.251
116.77.75.183
119.147.183.43
120.198.197.87
120.198.197.88
120.241.57.161
120.241.57.252
120.241.57.30
120.241.57.48
120.241.57.57
120.241.57.92
121.11.81.100
121.11.81.101
121.11.81.102
121.11.81.103
121.11.81.104
121.11.81.237
125.90.206.240
125.90.206.241
125.90.206.242
125.90.206.243
125.90.206.244
125.90.206.245
125.90.206.246
125.90.206.248
125.90.206.42
125.90.206.43
125.90.206.77
14.17.80.68
14.17.80.71
14.17.80.73
125.90.206.79
125.90.206.76
125.90.206.49
125.90.206.78
125.90.206.80
14.17.80.253
125.90.206.81
14.17.80.69
14.17.80.67
14.17.80.70
125.90.206.82
14.17.80.74
14.17.80.72
14.17.80.254
14.17.80.76
14.17.80.75
14.17.80.77
14.17.80.78
14.17.80.79
14.17.80.80
14.17.80.81
14.17.80.82
14.17.80.83
14.17.80.84
14.17.80.85
14.17.80.86
14.17.80.88
14.18.17.239
14.18.17.240
14.18.17.249
14.18.17.241
14.21.78.112
14.21.78.138
14.21.78.43
14.21.78.44
14.21.78.45
14.21.78.46
14.21.78.47
157.255.76.25
14.21.78.48
157.255.76.26
157.255.76.30
157.255.76.48
163.177.132.254
163.177.132.27
163.177.132.28
163.177.132.29
163.177.132.30
163.177.132.31
163.177.243.10
163.177.243.240
163.177.243.247
183.236.28.116
183.236.28.121
183.236.28.122
183.236.28.123
183.236.28.124
183.236.28.146
183.236.28.151
183.236.28.25
183.236.28.27
183.236.28.48
183.240.52.106
183.240.52.107
183.240.52.108
183.240.52.109
183.240.52.132
183.47.216.18
183.47.216.19
183.47.216.20
183.47.216.35
183.47.216.36
183.47.216.37
183.47.216.87
183.56.172.251
183.56.172.30
183.56.172.31
183.56.172.32
183.56.172.33
183.56.172.34
183.56.172.35
183.62.114.154
183.62.114.155
183.62.114.195
210.38.3.23
210.38.3.24
210.38.3.42
210.38.3.49
210.38.3.50
210.38.3.60
218.13.52.109
61.145.100.15
61.145.100.17
61.145.100.20
61.145.100.27
123.128.14.71
123.53.139.37
61.145.100.30
221.235.187.106
61.145.100.33
61.145.100.44
61.145.100.45
61.145.100.54
113.16.208.143
113.16.208.233
113.16.208.251
113.16.208.77
113.16.208.78
113.16.212.21
113.16.212.251
113.16.212.48
113.16.212.49
222.218.87.247
222.218.87.252
222.218.87.26
222.218.87.27
222.218.87.28
222.218.87.29
36.159.115.250
36.159.115.88
36.159.115.89
110.242.21.24
110.242.21.23
110.242.21.254
110.242.21.243
110.242.21.41
110.242.21.71
110.242.21.70
111.62.194.141
111.62.194.30
111.62.194.31
111.62.194.254
111.62.89.38
111.62.92.55
111.62.92.56
111.62.92.59
111.62.92.60
111.62.92.6
111.62.92.61
111.63.72.55
111.63.72.126
121.22.247.202
121.22.247.204
121.22.247.254
123.183.164.34
111.63.72.56
121.22.247.203
123.183.164.248
123.183.164.249
123.183.164.35
123.183.164.79
123.183.164.80
123.183.164.251
123.183.164.81
123.183.164.82
123.183.164.91
123.183.164.94
123.183.164.92
123.183.164.96
123.183.164.97
123.183.164.99
124.236.28.100
124.236.28.252
124.236.28.67
124.236.28.247
124.236.28.69
124.236.28.68
124.236.28.91
124.236.28.93
124.236.28.92
124.236.28.94
124.236.28.95
124.236.28.96
124.236.28.97
124.236.28.98
124.236.28.99
124.236.97.38
124.236.97.45
124.236.97.39
124.236.97.48
124.236.97.49
124.239.182.115
124.239.182.116
124.239.182.117
124.239.182.118
124.239.182.119
124.239.182.120
124.239.182.95
175.188.163.55
175.188.163.8
218.12.228.202
218.12.228.203
218.12.228.204
218.12.228.205
218.12.228.221
218.12.228.224
218.12.228.246
218.12.228.38
218.12.228.39
218.12.228.40
221.194.180.26
221.194.180.54
221.194.180.92
221.194.180.95
221.194.180.93
60.9.0.14
60.9.0.16
60.9.0.17
60.9.0.18
60.9.0.19
60.9.0.20
60.9.0.21
60.9.0.22
60.9.0.23
60.9.0.252
60.9.0.254
111.6.176.208
111.6.176.209
111.6.176.248
111.6.176.25
111.6.176.94
111.6.176.95
111.6.176.97
111.6.177.215
111.6.177.254
115.54.16.151
111.6.177.216
115.54.16.249
115.54.16.62
115.54.16.252
115.54.16.86
115.54.16.245
115.54.16.87
115.54.16.88
115.54.16.91
115.54.16.92
123.53.139.252
123.53.139.253
123.53.139.36
123.53.139.38
123.53.139.39
123.53.139.59
123.53.139.60
125.42.203.62
218.29.198.42
218.29.198.43
218.29.198.62
218.29.50.27
219.157.114.149
219.157.114.150
219.157.114.252
219.157.114.70
219.157.114.71
219.157.114.72
219.157.114.73
221.15.67.155
221.15.67.156
221.15.67.157
221.15.67.158
221.15.67.161
221.15.67.162
221.15.67.163
221.15.67.164
221.15.67.231
221.15.67.234
221.15.67.248
222.138.255.252
222.138.255.250
222.138.255.41
222.138.255.45
222.138.255.46
222.138.255.47
61.136.107.45
61.136.107.46
61.136.107.57
61.54.7.158
61.54.7.174
61.54.7.179
61.54.7.243
111.40.183.249
111.40.183.55
111.40.183.56
111.40.183.65
111.41.54.102
111.41.54.103
111.41.54.254
111.41.54.80
113.5.80.102
113.5.80.251
113.5.80.33
118.203.202.206
118.203.202.207
118.203.202.208
118.203.202.209
118.203.202.222
219.147.93.104
219.147.93.105
219.147.93.242
219.147.93.244
219.147.93.243
219.147.93.62
221.206.126.108
221.206.126.110
221.206.126.111
221.206.126.227
221.206.126.228
221.206.126.245
61.167.54.236
61.167.54.242
61.167.54.26
61.167.54.31
61.167.54.55
61.167.54.57
111.178.233.197
111.178.233.207
111.178.233.220
111.178.233.221
111.47.220.251
111.47.220.66
111.47.220.67
115.156.188.243
115.156.188.247
115.157.63.19
115.157.63.49
115.157.63.50
115.157.63.51
115.157.63.52
115.157.63.62
116.207.132.181
116.207.132.183
116.207.132.184
116.207.132.253
119.36.60.253
119.36.60.29
119.36.60.30
119.36.91.138
119.36.91.139
119.36.91.182
202.114.51.32
202.114.51.35
202.114.51.59
202.114.51.60
202.114.51.65
202.114.51.75
219.138.186.254
219.138.186.41
219.138.186.42
219.138.186.43
219.138.186.44
219.138.26.195
219.138.26.196
219.138.26.197
219.138.26.214
219.138.27.107
219.138.27.108
219.138.27.249
219.138.27.28
219.138.27.30
219.138.27.31
221.235.187.129
221.235.187.130
221.235.187.131
221.235.187.132
221.235.187.133
221.235.187.134
221.235.187.220
221.235.187.244
221.235.187.65
221.235.187.66
221.235.187.90
221.235.187.98
222.20.147.143
222.20.147.144
222.20.147.145
222.20.147.150
222.20.147.158
222.20.147.164
222.20.147.169
222.20.147.174
222.20.147.183
222.20.147.189
222.20.147.184
222.20.147.198
222.20.147.200
222.20.147.201
49.210.3.156
49.210.3.160
222.20.147.199
49.210.3.177
49.210.3.213
58.51.168.202
58.51.168.196
58.51.168.206
49.210.3.164
58.51.168.45
58.51.168.46
58.51.168.47
58.51.168.62
58.51.168.63
58.51.168.64
58.51.168.65
58.51.168.74
61.136.167.17
61.136.167.18
61.136.167.20
61.136.167.19
61.136.167.21
61.136.167.22
61.136.167.254
61.184.117.22
61.184.117.25
61.184.117.24
61.184.117.23
61.184.117.27
61.184.118.86
61.184.118.84
61.184.117.50
61.184.117.29
120.226.48.144
120.226.48.28
120.226.48.25
120.226.48.27
120.226.48.26
120.226.49.146
120.226.49.252
120.226.49.80
120.226.49.87
120.226.49.81
120.226.49.83
120.226.49.82
120.226.55.144
120.226.55.151
120.226.55.254
183.214.1.227
183.214.1.24
183.214.1.25
183.214.1.251
183.214.1.26
183.214.1.30
183.214.1.253
183.214.1.32
183.214.132.106
183.214.132.120
183.214.132.16
183.214.132.17
183.214.132.185
183.214.132.186
183.214.132.251
183.214.132.187
183.214.140.203
183.214.140.204
218.75.154.40
218.75.154.89
183.214.140.238
218.75.154.94
218.76.105.108
218.76.105.109
218.76.105.252
218.76.129.173
218.76.129.174
218.76.129.176
218.76.129.175
58.20.179.253
218.76.129.252
58.20.179.31
58.20.179.74
58.20.179.73
106.41.0.37
106.41.0.44
106.41.0.45
106.41.0.46
106.41.0.47
106.41.0.61
106.41.0.62
111.26.107.19
111.26.157.7
119.52.120.138
119.52.120.139
119.52.120.140
119.52.120.144
119.52.120.145
119.52.120.146
122.136.46.119
122.136.46.120
122.136.46.126
122.136.46.66
139.209.49.140
139.209.49.138
139.209.49.144
139.209.49.151
139.209.49.152
139.209.49.153
222.163.194.230
222.163.194.27
222.163.194.28
222.163.194.29
222.163.194.30
222.163.202.172
222.163.202.241
222.163.202.212
36.104.132.251
36.104.132.48
36.104.132.49
112.25.81.118
112.25.81.67
112.25.81.68
112.25.81.69
112.84.104.162
112.84.104.163
112.84.104.19
112.84.104.52
114.236.140.125
114.236.140.253
114.236.88.254
114.236.88.45
114.236.88.46
114.236.88.47
114.236.88.48
153.101.208.254
153.101.208.90
153.101.208.92
153.101.208.91
153.101.208.93
153.99.174.201
153.99.174.212
153.99.235.112
153.99.235.217
153.99.235.91
180.97.178.141
180.97.178.142
180.97.178.146
180.97.178.147
180.97.178.163
180.97.178.164
180.97.178.165
180.97.178.166
180.97.178.172
180.97.178.228
180.97.178.229
180.97.178.230
180.97.178.253
180.97.180.164
180.97.180.165
180.97.180.196
218.92.209.12
218.92.209.249
218.92.209.41
218.92.209.42
221.230.141.170
221.230.141.171
221.230.141.172
221.230.141.174
221.230.141.224
222.186.141.132
222.186.141.135
222.186.141.141
222.186.141.142
222.186.141.143
222.186.141.145
222.186.141.146
222.186.141.162
222.186.141.165
222.186.141.166
222.186.141.178
222.186.141.186
222.186.145.254
222.186.145.84
222.186.145.85
222.186.145.86
223.111.156.173
222.186.145.87
223.111.156.174
223.111.156.175
223.111.156.176
223.111.156.188
223.111.156.202
223.111.156.210
223.111.156.216
223.111.156.249
223.111.18.161
223.111.18.162
223.111.18.163
223.111.18.217
223.111.19.124
223.111.19.126
223.111.19.81
223.111.19.90
223.111.19.91
223.111.196.125
223.111.196.48
223.111.196.49
223.111.198.252
223.111.198.65
223.111.198.75
223.111.203.126
223.111.203.35
223.111.203.36
223.111.22.170
223.111.22.249
223.113.14.126
223.113.14.16
223.113.14.17
223.113.14.215
223.113.14.216
223.113.14.230
36.156.73.145
36.156.73.146
36.156.73.157
36.156.73.158
36.156.73.220
36.156.73.254
58.216.109.105
58.216.109.106
58.216.109.107
58.216.109.108
58.216.109.109
58.216.109.110
58.216.109.187
58.216.109.192
58.216.109.95
58.216.109.96
58.216.109.97
58.216.109.99
58.216.110.36
58.216.21.149
58.216.21.21
58.216.21.22
58.216.21.247
58.216.21.250
58.216.21.65
58.216.22.17
58.216.22.20
58.216.22.22
58.216.22.56
58.216.23.100
58.216.23.102
58.220.220.115
58.220.220.114
58.220.220.251
58.220.220.118
58.220.220.119
58.220.220.117
58.220.220.252
58.220.71.253
58.220.71.51
58.220.71.52
58.220.71.62
58.220.71.63
58.220.71.64
58.220.71.65
58.220.71.66
58.220.71.67
58.221.28.245
58.221.28.51
58.221.28.52
58.221.28.53
58.221.28.54
58.221.78.186
58.221.78.229
58.221.78.231
58.221.78.42
58.221.78.45
58.221.78.46
58.221.78.47
58.221.78.48
58.221.78.56
58.221.78.57
58.221.78.70
58.223.164.229
58.223.164.231
58.223.164.232
58.223.164.233
58.223.164.234
58.223.164.247
58.223.166.116
58.223.166.117
58.223.166.118
58.223.166.119
58.223.166.120
58.223.166.121
58.223.166.122
58.223.166.253
59.83.232.17
59.83.232.18
59.83.232.50
61.147.210.193
61.147.210.195
61.147.226.185
61.147.210.242
61.147.226.47
61.147.226.49
61.147.227.102
61.147.227.126
61.147.227.233
61.147.227.52
61.147.227.54
61.147.227.55
61.147.228.115
61.147.228.117
61.147.228.116
61.147.228.118
61.147.228.201
61.147.228.60
61.147.228.248
61.147.228.61
61.147.228.78
61.147.228.89
61.160.209.254
61.160.209.26
61.160.209.98
61.160.209.96
113.194.59.80
113.194.59.199
113.194.59.81
117.169.93.247
117.169.93.249
117.169.93.85
117.21.217.26
117.169.93.86
117.21.217.27
117.21.217.98
118.212.138.106
118.212.138.107
118.212.138.215
118.212.150.100
118.212.150.101
118.212.150.102
120.206.189.252
120.206.189.253
120.206.189.52
120.206.189.53
182.108.171.149
182.108.171.150
182.108.171.170
182.108.171.171
182.108.171.172
182.108.171.250
182.108.171.251
183.216.176.116
183.216.176.248
183.216.176.251
183.216.176.63
183.216.176.64
183.216.176.65
183.216.176.74
183.216.176.75
183.216.176.83
183.216.176.89
219.220.29.175
219.220.29.177
219.220.29.182
219.220.29.189
219.220.29.190
59.63.221.250
59.63.221.253
59.63.221.35
59.63.221.44
59.63.221.45
59.63.221.46
59.63.221.47
59.63.221.56
59.63.241.54
59.63.241.81
59.63.241.82
59.63.241.83
59.63.241.84
59.63.241.85
101.246.182.157
101.246.182.160
101.246.182.161
101.246.182.209
101.246.182.214
101.246.182.219
101.246.182.251
123.184.108.251
123.184.108.60
123.184.108.61
124.95.148.124
124.95.148.254
124.95.148.30
124.95.148.29
124.95.148.31
182.201.212.140
182.201.212.141
182.201.212.142
182.201.212.242
182.201.212.79
218.60.185.100
218.60.185.102
218.60.185.103
218.60.185.215
218.60.185.248
218.60.185.251
218.60.185.46
218.60.185.47
218.60.185.48
221.180.192.100
221.180.192.131
221.180.192.235
221.180.192.254
221.180.192.39
221.180.192.49
221.180.192.61
221.180.208.190
221.180.208.214
221.180.208.252
221.180.208.46
221.180.208.47
221.180.208.48
221.180.208.50
221.180.208.54
221.180.218.248
221.180.218.73
221.180.218.74
42.176.192.171
42.176.192.226
42.176.192.76
42.176.192.77
42.176.192.78
59.44.25.58
59.44.25.8
59.44.30.121
59.44.30.25
59.44.30.26
59.44.30.27
59.44.30.54
59.47.227.144
59.47.227.146
59.47.227.148
59.47.227.149
60.18.86.100
60.18.86.101
60.18.86.253
60.18.86.91
61.243.144.141
61.243.144.142
61.243.144.244
106.40.140.224
106.40.140.222
110.18.244.134
106.40.140.254
110.18.244.13
110.18.244.252
110.18.244.44
110.18.244.63
110.18.244.76
110.19.204.219
110.19.204.220
110.19.204.221
110.19.204.222
110.19.204.254
117.161.19.126
117.161.19.19
117.161.19.22
117.161.66.125
117.161.66.71
218.21.175.24
222.74.113.196
222.74.113.219
222.74.113.228
222.74.113.230
222.74.113.231
222.74.113.229
36.102.230.136
36.102.230.137
36.102.230.138
36.102.230.19
36.102.230.20
36.102.230.21
36.102.230.22
36.102.230.253
36.102.230.254
58.18.254.207
58.18.254.221
58.18.254.253
120.253.100.20
120.253.100.21
120.253.100.22
120.253.100.23
120.253.100.30
211.138.60.159
211.138.60.189
112.240.60.172
112.240.60.208
112.240.60.209
112.240.60.210
112.240.60.211
112.240.60.212
112.240.60.213
112.240.60.214
112.240.60.215
112.240.60.216
112.240.60.217
112.240.60.222
112.240.60.236
112.240.60.253
112.240.60.69
112.240.60.72
112.240.60.73
112.240.60.75
112.253.38.37
112.240.60.88
119.176.61.23
119.176.61.26
119.176.61.99
120.220.18.112
120.220.18.113
120.220.18.252
120.221.23.39
120.221.23.62
120.221.23.63
120.221.23.64
120.221.64.160
120.221.64.161
120.221.64.169
120.221.64.251
120.221.64.53
120.221.64.54
120.221.64.55
120.221.64.77
123.128.14.254
123.128.14.69
123.128.14.68
123.128.14.49
123.128.14.70
123.128.14.72
150.138.111.251
150.138.111.31
150.138.111.32
150.138.111.33
150.138.167.234
150.138.167.50
150.138.167.52
150.138.169.120
150.138.169.121
150.138.169.123
150.138.169.124
150.138.169.136
150.138.169.137
150.138.169.238
150.138.214.122
150.138.214.84
150.138.214.124
150.138.214.85
150.138.214.86
182.34.127.22
182.34.127.21
182.34.127.253
182.34.127.23
182.34.127.45
182.34.127.48
218.58.206.37
218.58.206.39
218.58.206.40
218.58.206.41
218.58.206.43
218.58.206.46
218.58.206.49
218.58.206.53
218.58.206.55
218.58.206.57
218.58.206.78
27.195.145.121
27.195.145.123
27.195.145.124
27.195.145.249
27.195.145.52
27.195.145.62
27.195.145.61
59.80.28.254
59.80.28.58
59.80.29.21
59.80.29.20
59.80.29.22
59.80.29.30
59.80.29.40
59.80.29.33
59.83.211.215
59.83.228.160
59.83.228.88
59.83.228.89
60.210.21.85
60.210.21.86
60.210.21.87
60.210.21.88
60.210.21.89
60.210.23.175
60.210.23.177
60.210.23.178
60.210.23.23
60.210.23.25
60.210.23.27
60.213.21.117
60.213.21.118
60.213.21.156
60.213.21.157
60.213.21.214
60.213.21.244
60.213.21.243
60.213.21.245
60.213.21.252
60.213.22.34
60.213.22.57
61.156.243.108
61.156.243.109
61.156.243.110
61.156.243.111
61.156.243.112
61.156.243.246
61.162.100.102
61.162.100.103
61.162.100.105
61.162.100.107
61.162.100.252
61.162.100.41
61.162.100.44
124.165.125.126
124.165.125.13
124.165.125.134
124.165.125.253
124.167.218.116
124.167.218.251
124.167.218.117
183.201.225.249
183.201.225.74
183.201.225.73
218.26.75.149
218.26.75.150
218.26.75.151
218.26.75.152
218.26.75.153
218.26.75.206
218.26.75.236
59.49.89.100
59.49.89.45
59.49.89.69
60.220.196.220
60.220.196.221
60.220.196.222
60.220.196.223
60.220.196.235
60.220.196.250
60.220.196.240
60.222.200.71
111.19.215.40
111.19.215.39
111.19.215.127
111.19.215.41
111.19.215.69
111.19.233.252
111.19.233.253
111.19.233.59
111.19.233.60
111.19.233.99
113.142.80.223
111.19.233.87
113.142.80.69
113.142.80.71
113.142.80.72
113.142.88.253
113.142.88.31
113.142.90.226
113.142.90.235
113.142.90.236
117.23.2.241
117.23.2.252
117.23.2.28
117.23.2.29
117.23.2.30
123.138.157.85
123.138.157.122
123.138.157.93
123.138.157.96
123.138.203.115
123.138.203.20
123.138.203.21
123.138.203.56
123.138.60.154
124.116.133.15
124.116.133.126
124.116.133.16
124.116.133.17
124.116.133.18
124.116.133.19
124.116.133.20
124.116.133.63
124.116.133.76
219.145.171.249
124.116.133.77
219.145.171.40
219.145.171.41
222.24.122.79
222.24.122.86
222.24.122.89
101.227.99.21
101.227.99.22
101.227.99.233
121.46.247.247
121.46.247.73
121.46.247.74
121.46.248.79
124.14.20.219
180.169.63.51
180.169.63.52
180.169.63.59
180.169.63.60
211.144.81.22
211.144.81.24
218.242.102.131
218.242.102.137
218.242.102.138
218.242.102.139
218.242.102.151
218.78.185.251
218.78.185.34
218.78.185.35
222.44.151.191
222.44.151.25
61.151.238.197
61.151.238.222
61.151.238.229
117.174.21.130
117.174.21.135
117.174.21.238
117.176.232.253
117.176.232.83
117.176.232.84
118.118.216.129
118.118.216.130
118.118.216.131
118.118.216.252
118.118.216.46
118.123.233.14
118.123.233.15
118.123.233.252
118.123.233.254
118.123.233.30
118.123.233.31
118.123.233.32
118.123.237.174
118.123.237.176
118.123.237.245
118.123.237.254
118.123.237.27
118.123.237.28
118.123.237.29
118.123.251.19
118.123.251.247
118.123.251.254
118.123.251.73
118.123.251.76
118.123.251.77
125.64.102.254
125.64.102.69
125.64.102.71
125.66.85.108
125.66.85.31
125.66.85.30
182.136.72.130
182.136.72.131
182.140.147.54
182.140.218.23
182.140.218.24
182.140.218.25
182.140.218.254
182.140.218.26
182.140.236.124
182.140.236.57
182.140.236.58
183.222.97.163
183.222.97.164
183.222.97.165
183.222.97.166
183.222.97.246
183.222.97.254
202.98.156.253
202.98.156.59
202.98.156.60
202.98.156.61
211.162.189.192
211.162.189.193
211.162.189.194
211.162.189.207
211.162.189.223
223.86.219.254
223.86.219.65
223.86.219.66
60.255.143.93
60.255.143.94
60.255.143.95
60.255.143.96
61.157.124.252
61.157.124.38
61.157.124.40
61.157.124.42
61.157.124.47
61.157.124.48
61.188.191.254
61.188.191.34
61.188.191.81
61.188.191.82
61.188.191.83
111.161.122.100
111.161.122.128
111.161.122.133
111.161.122.134
111.161.122.135
111.161.122.137
111.161.122.181
111.161.122.240
111.161.122.67
111.161.22.15
111.161.22.16
111.161.22.17
111.161.22.18
111.161.22.19
111.161.22.62
125.39.1.134
125.39.1.191
42.81.144.179
42.81.144.180
42.81.144.191
42.81.144.23
42.81.144.31
42.81.144.39
42.81.144.53
42.81.144.54
42.81.144.55
42.81.144.56
42.81.144.77
42.81.144.78
42.81.144.79
60.28.100.155
60.28.100.156
60.28.100.157
60.28.100.158
60.28.100.248
117.180.229.254
117.180.229.142
117.145.179.22
117.145.179.20
117.145.179.19
117.145.179.248
106.57.180.253
106.57.180.62
106.57.180.63
14.204.185.100
14.204.185.101
14.204.185.102
14.204.185.123
14.204.185.254
182.242.50.245
182.242.50.46
14.204.185.91
182.242.50.48
220.165.142.137
220.165.142.253
220.165.142.9
222.221.102.254
222.221.102.39
39.130.253.226
222.221.102.40
39.130.253.227
39.130.253.253
39.130.253.45
39.130.253.46
39.130.253.48
39.130.253.52
39.130.253.53
101.69.104.20
101.69.104.21
101.69.104.29
101.69.104.30
101.69.104.63
101.69.104.74
101.69.146.234
101.69.146.30
101.69.146.31
101.69.146.32
101.69.146.33
101.69.146.35
101.69.146.34
101.69.146.41
111.0.23.138
111.0.22.116
111.0.23.139
112.16.227.101
112.16.227.102
112.16.227.103
112.16.227.104
112.16.227.125
112.16.227.254
112.17.175.197
112.17.27.254
112.17.27.55
112.17.53.253
112.17.53.97
112.17.53.98
113.215.16.58
115.223.24.254
113.215.16.59
115.223.24.80
115.223.24.81
115.223.24.82
115.223.24.83
115.223.7.123
115.223.7.125
115.223.7.124
115.223.7.126
115.223.7.127
115.223.7.128
115.223.7.173
115.223.7.174
115.223.7.175
115.223.7.176
115.223.7.248
115.223.7.249
117.148.128.68
117.148.128.93
117.148.128.96
117.148.128.97
117.148.128.98
117.149.154.185
117.149.154.186
117.149.154.187
117.149.154.188
117.149.154.237
117.149.154.250
117.149.154.251
117.149.155.148
117.149.155.147
117.149.155.49
117.149.155.76
122.224.186.221
122.224.186.222
122.224.186.224
122.224.186.225
122.224.186.226
122.224.186.227
122.224.186.252
122.224.186.253
122.225.28.159
122.225.28.160
122.225.28.161
122.225.28.163
122.225.28.251
122.225.83.25
122.225.83.28
122.228.237.74
122.228.237.75
122.228.237.76
122.228.239.233
122.228.239.234
122.228.239.235
122.228.239.236
122.228.239.238
122.228.239.243
122.228.239.244
122.228.239.246
122.228.24.103
122.228.24.105
122.228.24.106
122.228.24.107
122.228.24.114
122.228.24.173
122.228.6.197
122.228.6.200
122.228.6.201
122.228.6.209
122.228.6.217
122.228.6.220
122.228.6.218
183.131.124.249
183.131.124.38
183.131.124.39
183.131.124.40
183.131.124.43
183.131.124.58
183.131.124.59
183.131.124.60
183.131.124.61
183.131.124.62
183.131.168.146
183.131.168.147
183.131.168.148
183.131.168.187
183.131.171.42
183.131.171.47
183.131.26.41
183.131.26.42
183.131.26.43
183.131.26.44
183.131.26.45
183.131.26.46
183.131.26.47
183.131.26.5
183.131.26.60
183.134.12.26
183.134.12.254
183.134.12.27
183.134.12.28
183.134.42.153
183.134.42.154
183.134.42.155
183.134.42.156
183.134.42.157
183.134.42.158
183.134.42.159
183.134.42.17
183.134.42.18
183.134.42.19
183.134.42.190
183.134.42.191
183.134.42.20
183.134.42.247
183.134.53.153
183.134.53.155
183.134.53.156
183.134.53.157
183.134.53.158
183.134.53.223
183.134.53.252
183.134.53.248
183.146.22.135
183.146.22.136
183.146.22.137
183.146.22.138
183.146.22.143
183.146.22.145
183.146.22.146
183.146.22.159
183.146.22.171
183.146.22.179
36.25.241.249
36.25.241.250
36.25.241.39
36.25.241.40
36.25.241.41
36.25.241.75
36.25.241.76
36.25.241.77
113.207.0.17
113.207.0.97
113.207.0.98
113.207.10.204
113.207.10.218
113.207.6.210
113.207.6.211
113.207.6.212
113.207.6.213
113.207.70.194
113.207.70.207
113.207.70.208
113.207.72.102
113.207.72.103
113.207.70.209
113.207.77.117
113.207.77.118
113.207.77.121
113.207.77.247
113.207.77.252
113.207.79.17
113.207.79.34
113.207.79.42
113.207.79.37
113.207.81.126
113.207.81.76
113.207.81.77
113.207.81.78
113.207.81.81
119.84.128.19
119.84.128.21
119.84.128.22
119.84.128.254
183.66.108.204
183.66.108.205
183.66.109.253
183.66.109.254
183.66.109.38
183.66.109.44
183.66.109.45
183.66.109.50
211.162.212.156
211.162.212.210
211.162.212.211
211.162.212.212
211.162.212.213
211.162.212.246
221.178.0.252
================================================
FILE: config/AutoSynchroTime.py
================================================
# coding=utf-8
import os
import platform
import ntplib
import datetime
def autoSynchroTime():
"""
同步北京时间,执行时候,请务必用sudo,sudo,sudo 执行,否则会报权限错误,windows打开ide或者cmd请用管理员身份
:return:
"""
c = ntplib.NTPClient()
hosts = ['ntp1.aliyun.com', 'ntp2.aliyun.com', 'ntp3.aliyun.com', 'ntp4.aliyun.com', 'cn.pool.ntp.org']
print(u"正在同步时间,请耐心等待30秒左右,如果下面有错误发送,可以忽略!!")
print(u"系统当前时间{}".format(str(datetime.datetime.now())[:22]))
system = platform.system()
if system == "Windows": # windows 同步时间未测试过,参考地址:https://www.jianshu.com/p/92ec15da6cc3
for host in hosts:
os.popen('w32tm /register')
os.popen('net start w32time')
os.popen('w32tm /config /manualpeerlist:"{}" /syncfromflags:manual /reliable:yes /update'.format(host))
os.popen('ping -n 3 127.0.0.1 >nul')
sin = os.popen('w32tm /resync')
if sin is 0:
break
else: # mac同步地址,如果ntpdate未安装,brew install ntpdate linux 安装 yum install -y ntpdate
for host in hosts:
sin = os.popen('ntpdate {}'.format(host))
if sin is 0:
break
print(u"同步后时间:{}".format(str(datetime.datetime.now())[:22]))
if __name__ == '__main__':
autoSynchroTime()
================================================
FILE: config/TicketEnmu.py
================================================
# coding=utf-8
from enum import Enum
class ticket(object):
QUERY_C = u"查询到有余票,尝试提交订单"
QUERY_IN_BLACK_LIST = u"该车次{} 正在被关小黑屋,跳过此车次"
SUCCESS_CODE = 000000
FAIL_CODE = 999999
AUTO_SUBMIT_ORDER_REQUEST_C = u"提交订单成功"
AUTO_SUBMIT_ORDER_REQUEST_F = u"提交订单失败,重新刷票中"
AUTO_SUBMIT_NEED_CODE = u"需要验证码"
AUTO_SUBMIT_NOT_NEED_CODE = u"不需要验证码"
TICKET_BLACK_LIST_TIME = 5 # 加入小黑屋的等待时间,默认5 min
DTO_NOT_FOUND = u"未查找到常用联系人, 请查证后添加!!"
DTO_NOT_IN_LIST = u"联系人不在列表中,请查证后添加!!"
QUEUE_TICKET_SHORT = u"当前余票数小于乘车人数,放弃订票"
QUEUE_TICKET_SUCCESS = u"排队成功, 当前余票还剩余: {0}张"
QUEUE_JOIN_BLACK = u"排队发现未知错误{0},将此列车 {1}加入小黑屋"
QUEUE_WARNING_MSG = u"排队异常,错误信息:{0}, 将此列车 {1}加入小黑屋"
OUT_NUM = 120 # 排队请求12306的次数
WAIT_OUT_NUM = u"超出排队时间,自动放弃,正在重新刷票"
WAIT_ORDER_SUCCESS = u"恭喜您订票成功,订单号为:{0}, 请立即打开浏览器登录12306,访问‘未完成订单’,在30分钟内完成支付!"
WAIT_AFTER_NATE_SUCCESS = u"候补订单已完成,请立即打开浏览器登录12306,访问‘候补订单’,在30分钟内完成支付!"
WAIT_ORDER_CONTINUE = u"排队等待时间预计还剩 {0} ms"
WAIT_ORDER_FAIL = u"排队等待失败,错误消息:{0}"
WAIT_ORDER_NUM = u"第{0}次排队中,请耐心等待"
WAIT_ORDER_SUB_FAIL = u"订单提交失败!,正在重新刷票"
CANCEL_ORDER_SUCCESS = u"排队超时,已为您自动取消订单,订单编号: {0}"
CANCEL_ORDER_FAIL = u"排队超时,取消订单失败, 订单号{0}"
REST_TIME = u"12306休息时间,本程序自动停止,明天早上6点将自动运行"
REST_TIME_PAST = u"休息时间已过,重新开启检票功能"
LOGIN_SESSION_FAIL = u"用户检查失败:{0},可能未登录,可能session已经失效, 正在重新登录中"
================================================
FILE: config/__init__.py
================================================
================================================
FILE: config/configCommon.py
================================================
# -*- coding: utf-8 -*-
import datetime
import os
import random
import sys
import time
from myException.ticketConfigException import ticketConfigException
rushRefreshMinTimeIntval = 2000
rushRefreshMaxTimeIntval = 3600000
rushRefreshTimeIntval = 100
# 最早运行时间
maxRunTime = 6
# 程序停止时间
maxRunStopTime = 23
# 可售天数
maxDate = 29
RS_SUC = 0
RS_TIMEOUT = 1
RS_JSON_ERROR = 2
RS_OTHER_ERROR = 3
seat_conf = {'商务座': 32,
'一等座': 31,
'二等座': 30,
'特等座': 25,
'软卧': 23,
'硬卧': 28,
'软座': 24,
'硬座': 29,
'无座': 26,
'动卧': 33,
}
if sys.version_info.major == 2:
seat_conf_2 = dict([(v, k) for (k, v) in seat_conf.iteritems()])
else:
seat_conf_2 = dict([(v, k) for (k, v) in seat_conf.items()])
def getNowTimestamp():
return time.time()
def decMakeDir(func):
def handleFunc(*args, **kwargs):
dirname = func(*args, **kwargs)
if not os.path.exists(dirname):
os.makedirs(dirname)
elif not os.path.isdir(dirname):
pass
return dirname
return func
def getWorkDir():
return os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
#
# def fileOpen(path):
# """
# 文件读取兼容2和3
# :param path: 文件读取路径
# :return:
# """
# try:
# with open(path, "r", ) as f:
# return f
# except TypeError:
# with open(path, "r", ) as f:
# return f
@decMakeDir
def getTmpDir():
return os.path.join(getWorkDir(), "tmp")
@decMakeDir
def getLogDir():
return os.path.join(getTmpDir(), "log")
@decMakeDir
def getCacheDir():
return os.path.join(getTmpDir(), "cache")
@decMakeDir
def getVCodeDir():
return os.path.join(getTmpDir(), "vcode")
def getVCodeImageFile(imageName):
return os.path.join(getVCodeDir(), imageName + ".jpg")
def getCacheFile(cacheType):
return os.path.join(getCacheDir(), cacheType + ".cache")
def checkSleepTime(session):
now = datetime.datetime.now()
if now.hour >= maxRunStopTime or now.hour < maxRunTime:
print(u"12306休息时间,本程序自动停止,明天早上六点将自动运行")
open_time = datetime.datetime(now.year, now.month, now.day, maxRunTime)
if open_time < now:
open_time += datetime.timedelta(1)
time.sleep((open_time - now).seconds + round(random.uniform(1, 10)))
session.call_login()
def checkDate(station_dates):
"""
检查日期是否合法
:param station_dates:
:return:
"""
today = datetime.datetime.now()
maxDay = (today + datetime.timedelta(maxDate)).strftime("%Y-%m-%d")
for station_date in station_dates[::-1]:
date = datetime.datetime.strftime(datetime.datetime.strptime(station_date, "%Y-%m-%d"), "%Y-%m-%d")
if date < today.strftime("%Y-%m-%d") or date > maxDay:
print(u"警告:当前时间配置有小于当前时间或者大于最大时间: {}, 已自动忽略".format(station_date))
station_dates.remove(station_date)
if not station_dates:
print(u"当前日期设置无符合查询条件的,已被全部删除,请查证后添加!!!")
raise ticketConfigException(u"当前日期设置无符合查询条件的,已被全部删除,请查证后添加!!!")
else:
station_dates[station_dates.index(station_date)] = date
return station_dates
================================================
FILE: config/emailConf.py
================================================
# -*- coding: utf8 -*-
import socket
__author__ = 'MR.wen'
import TickerConfig
from email.header import Header
from email.mime.text import MIMEText
import smtplib
def sendEmail(msg):
"""
邮件通知
:param str: email content
:return:
"""
try:
if TickerConfig.EMAIL_CONF["IS_MAIL"]:
sender = TickerConfig.EMAIL_CONF["email"]
receiver = TickerConfig.EMAIL_CONF["notice_email_list"]
subject = '恭喜,您已订票成功'
username = TickerConfig.EMAIL_CONF["username"]
password = TickerConfig.EMAIL_CONF["password"]
host = TickerConfig.EMAIL_CONF["host"]
s = "{0}".format(msg)
msg = MIMEText(s, 'plain', 'utf-8') # 中文需参数‘utf-8’,单字节字符不需要
msg['Subject'] = Header(subject, 'utf-8')
msg['From'] = sender
msg['To'] = receiver
try:
smtp = smtplib.SMTP_SSL(host)
smtp.connect(host)
except socket.error:
smtp = smtplib.SMTP()
smtp.connect(host)
smtp.connect(host)
smtp.login(username, password)
smtp.sendmail(sender, receiver.split(","), msg.as_string())
smtp.quit()
print(u"邮件已通知, 请查收")
except Exception as e:
print(u"邮件配置有误{}".format(e))
if __name__ == '__main__':
sendEmail(1)
================================================
FILE: config/getCookie.py
================================================
import json
import random
import re
import time
import os
import TickerConfig
from config.urlConf import urls
def getDrvicesID(session):
"""
:return:
"""
print("cookie获取中")
if TickerConfig.COOKIE_TYPE is 1:
from selenium import webdriver
cookies = []
# 解决放镜像里 DevToolsActivePort file doesn't exist的问题
options = webdriver.ChromeOptions()
if os.name != 'nt' and TickerConfig.CHROME_CHROME_PATH:
options = webdriver.ChromeOptions()
options.binary_location = TickerConfig.CHROME_CHROME_PATH
options.add_argument(
'--user-agent=Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/79.0.3945.88 Safari/537.36')
options.add_argument("--no-sandbox")
options.add_argument("--headless")
driver = webdriver.Chrome(executable_path=TickerConfig.CHROME_PATH,chrome_options=options)
driver.get("https://www.12306.cn/index/index.html")
time.sleep(10)
for c in driver.get_cookies():
cookie = dict()
print()
if c.get("name") == "RAIL_DEVICEID" or c.get("name") == "RAIL_EXPIRATION":
cookie[c.get("name")] = c.get("value")
cookies.append(cookie)
print(f"获取cookie: {cookies}")
if cookies:
session.httpClint.set_cookies(cookies)
session.cookies = cookies
print("cookie获取完成")
elif TickerConfig.COOKIE_TYPE is 2:
request_device_id(session)
elif TickerConfig.COOKIE_TYPE is 3:
# RAIL_DEVICEID,RAIL_EXPIRATION的值打开12306官网可以获取headers-Cookies
if not TickerConfig.RAIL_DEVICEID or not TickerConfig.RAIL_EXPIRATION:
print("警告!!: RAIL_DEVICEID,RAIL_EXPIRATION的值为空,请手动打开12306官网可以获取headers-Cookies中的RAIL_DEVICEID,RAIL_EXPIRATION,填入配置文件中")
cookies = [{
"RAIL_DEVICEID": TickerConfig.RAIL_DEVICEID,
"RAIL_EXPIRATION": TickerConfig.RAIL_EXPIRATION,
}]
session.httpClint.set_cookies(cookies)
session.cookies = cookies
def request_device_id(session):
"""
获取加密后的浏览器特征 ID
:return:
"""
params = {"algID": request_alg_id(session), "timestamp": int(time.time() * 1000)}
params = dict(params, **_get_hash_code_params())
response = session.httpClint.send(urls.get("getDevicesId"), params=params)
if response.find('callbackFunction') >= 0:
result = response[18:-2]
try:
result = json.loads(result)
session.httpClint.set_cookies([{
'RAIL_EXPIRATION': result.get('exp'),
'RAIL_DEVICEID': result.get('dfp'),
}])
session.cookies = [{
'RAIL_EXPIRATION': result.get('exp'),
'RAIL_DEVICEID': result.get('dfp'),
}]
except:
return False
def request_alg_id(session):
response = session.httpClint.send(urls.get("GetJS"))
result = re.search(r'algID\\x3d(.*?)\\x26', response)
try:
return result.group(1)
except (IndexError, AttributeError) as e:
pass
return ""
def _get_hash_code_params():
from collections import OrderedDict
data = {
'adblock': '0',
'browserLanguage': 'en-US',
'cookieEnabled': '1',
'custID': '133',
'doNotTrack': 'unknown',
'flashVersion': '0',
'javaEnabled': '0',
'jsFonts': 'c227b88b01f5c513710d4b9f16a5ce52',
'localCode': '3232236206',
'mimeTypes': '52d67b2a5aa5e031084733d5006cc664',
'os': 'MacIntel',
'platform': 'WEB',
'plugins': 'd22ca0b81584fbea62237b14bd04c866',
'scrAvailSize': str(random.randint(500, 1000)) + 'x1920',
'srcScreenSize': '24xx1080x1920',
'storeDb': 'i1l1o1s1',
'timeZone': '-8',
'touchSupport': '99115dfb07133750ba677d055874de87',
'userAgent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_6) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/75.0.' + str(
random.randint(
5000, 7000)) + '.0 Safari/537.36',
'webSmartID': 'f4e3b7b14cc647e30a6267028ad54c56',
}
data_trans = {
'browserVersion': 'd435',
'touchSupport': 'wNLf',
'systemLanguage': 'e6OK',
'scrWidth': 'ssI5',
'openDatabase': 'V8vl',
'scrAvailSize': 'TeRS',
'hasLiedResolution': '3neK',
'hasLiedOs': 'ci5c',
'timeZone': 'q5aJ',
'userAgent': '0aew',
'userLanguage': 'hLzX',
'jsFonts': 'EOQP',
'scrAvailHeight': '88tV',
'browserName': '-UVA',
'cookieCode': 'VySQ',
'online': '9vyE',
'scrAvailWidth': 'E-lJ',
'flashVersion': 'dzuS',
'scrDeviceXDPI': '3jCe',
'srcScreenSize': 'tOHY',
'storeDb': 'Fvje',
'doNotTrack': 'VEek',
'mimeTypes': 'jp76',
'sessionStorage': 'HVia',
'cookieEnabled': 'VPIf',
'os': 'hAqN',
'hasLiedLanguages': 'j5po',
'hasLiedBrowser': '2xC5',
'webSmartID': 'E3gR',
'appcodeName': 'qT7b',
'javaEnabled': 'yD16',
'plugins': 'ks0Q',
'appMinorVersion': 'qBVW',
'cpuClass': 'Md7A',
'indexedDb': '3sw-',
'adblock': 'FMQw',
'localCode': 'lEnu',
'browserLanguage': 'q4f3',
'scrHeight': '5Jwy',
'localStorage': 'XM7l',
'historyList': 'kU5z',
'scrColorDepth': "qmyu"
}
data = OrderedDict(data)
d = ''
params = {}
for key, item in data.items():
d += key + item
key = data_trans[key] if key in data_trans else key
params[key] = item
d_len = len(d)
d_f = int(d_len / 3) if d_len % 3 == 0 else int(d_len / 3) + 1
if d_len >= 3:
d = d[d_f:2 * d_f] + d[2 * d_f:d_len] + d[0: d_f]
d_len = len(d)
d_f = int(d_len / 3) if d_len % 3 == 0 else int(d_len / 3) + 1
if d_len >= 3:
d = d[2 * d_f:d_len] + d[0: d_f] + d[1 * d_f: 2 * d_f]
d = _encode_data_str_v2(d)
d = _encode_data_str_v2(d)
d = _encode_data_str_v2(d)
data_str = _encode_string(d)
params['hashCode'] = data_str
return params
def _encode_data_str_v2(d):
b = len(d)
if b % 2 == 0:
return d[b // 2: b] + d[0:b // 2]
else:
return d[b // 2 + 1:b] + d[b // 2] + d[0:b // 2]
def _encode_string(str):
import hashlib
import base64
result = base64.b64encode(hashlib.sha256(str.encode()).digest()).decode()
return result.replace('+', '-').replace('/', '_').replace('=', '')
================================================
FILE: config/logger.py
================================================
#coding: utf-8
import os
import time
import logging
from config import configCommon
logger = None
loggerHandler = None
dateStr = '' #默认拥有日期后缀
suffix = '' #除了日期外的后缀
def setSuffix(s):
global suffix
suffix = s
def getTodayDateStr():
return time.strftime("%Y-%m-%d", time.localtime(configCommon.getNowTimestamp()))
def setDateStr(s):
global dateStr
dateStr = s
def isAnotherDay(s):
global dateStr
return dateStr != s
def getLogFile():
global dateStr, suffix
rtn = os.path.join(configCommon.getLogDir(), dateStr)
if suffix:
rtn += "_" + suffix
return rtn + ".log"
def log(msg, func = "info"):
global logger
if not logger:
logger = logging.getLogger()
logger.setLevel(logging.INFO)
todayStr = getTodayDateStr()
if isAnotherDay(todayStr):
setDateStr(todayStr)
logger.removeHandler(loggerHandler)
fh = logging.FileHandler(getLogFile())
fm = logging.Formatter(u'[%(asctime)s][%(levelname)8s] --- %(message)s (%(filename)s:%(lineno)s)')
fh.setFormatter(fm)
logger.addHandler(fh)
levels = {
"debug": logger.debug,
"info": logger.info,
"warning": logger.warning,
"error": logger.error,
"critical": logger.critical
}
levels[func](msg)
================================================
FILE: config/pushbearConf.py
================================================
# -*- coding: utf8 -*-
import TickerConfig
from config.urlConf import urls
from myUrllib.httpUtils import HTTPClient
PUSH_BEAR_API_PATH = "https://pushbear.ftqq.com/sub"
def sendPushBear(msg):
"""
pushBear微信通知
:param str: 通知内容 content
:return:
"""
if TickerConfig.PUSHBEAR_CONF["is_pushbear"] and TickerConfig.PUSHBEAR_CONF["send_key"].strip() != "":
try:
sendPushBearUrls = urls.get("Pushbear")
data = {
"sendkey": TickerConfig.PUSHBEAR_CONF["send_key"].strip(),
"text": "易行购票成功通知",
"desp": msg
}
httpClint = HTTPClient(0)
sendPushBeaRsp = httpClint.send(sendPushBearUrls, data=data)
if sendPushBeaRsp.get("code") is 0:
print(u"已下发 pushbear 微信通知, 请查收")
else:
print(sendPushBeaRsp)
except Exception as e:
print(u"pushbear 配置有误 {}".format(e))
else:
pass
if __name__ == '__main__':
sendPushBear(1)
================================================
FILE: config/serverchanConf.py
================================================
# -*- coding: utf8 -*-
import TickerConfig
from config.urlConf import urls
from myUrllib.httpUtils import HTTPClient
PUSH_SERVER_CHAN_PATH = "https://sc.ftqq.com"
def sendServerChan(msg):
"""
pushBear微信通知
:param str: 通知内容 content
:return:
"""
if (
TickerConfig.SERVER_CHAN_CONF["is_server_chan"]
and TickerConfig.SERVER_CHAN_CONF["secret"].strip() != ""
):
try:
secret = TickerConfig.SERVER_CHAN_CONF["secret"].strip()
sendServerChanUrls = urls.get("ServerChan")
sendServerChanUrls["req_url"] += f'{secret}.send'
params = {"text": "易行购票成功通知", "desp": msg}
httpClint = HTTPClient(0)
sendServerChanRsp = httpClint.send(sendServerChanUrls, params=params)
if sendServerChanRsp.get("errno") == 0:
print(u"已下发 Server酱 微信通知, 请查收")
else:
print(sendServerChanRsp)
except Exception as e:
print(u"Server酱 配置有误 {}".format(e))
if __name__ == "__main__":
sendServerChan(1)
================================================
FILE: config/urlConf.py
================================================
# coding=utf-8
import random
import TickerConfig
import time
urls = {
"auth": { # 登录接口
"req_url": "/passport/web/auth/uamtk",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_logger": True,
"is_json": True,
"is_cdn": True,
},
"uamtk-static": { # 登录接口
"req_url": "/passport/web/auth/uamtk-static",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 3,
"s_time": 0.1,
"is_logger": True,
"is_json": True,
"is_cdn": True,
},
"login": { # 登录接口
"req_url": "/passport/web/login",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.5,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"left_ticket_init": { # 登录接口
"req_url": "/otn/leftTicket/init",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_logger": False,
"is_cdn": True,
"is_json": False,
},
"getCodeImg": { # 登录验证码
"req_url": "/passport/captcha/captcha-image?login_site=E&module=login&rand=sjrand&{0}",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_logger": False,
"is_json": False,
"is_cdn": True,
"not_decode": True,
},
"getCodeImg1": { # 登录验证码
"req_url": "/passport/captcha/captcha-image64?login_site=E&module=login&rand=sjrand&{0}&callback=jQuery19108016482864806321_1554298927290&_=1554298927293",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": False,
},
"codeCheck": { # 验证码校验
"req_url": "/passport/captcha/captcha-check",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": False,
},
"codeCheck1": { # 验证码校验
"req_url": "/passport/captcha/captcha-check?callback=jQuery19108016482864806321_1554298927290&answer={0}&rand=sjrand&login_site=E&_={1}",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/resources/login.html",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": False,
},
"loginInit": { # 登录页面
"req_url": "/otn/login/init",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/index/init",
"Host": "kyfw.12306.cn",
"re_try": 1,
"re_time": 1,
"s_time": 0.1,
"is_logger": False,
"is_cdn": True,
"is_json": False,
},
"loginInitCdn": { # 登录页面
"req_url": "/otn/login/init",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/index/init",
"Host": "kyfw.12306.cn",
"re_try": 1,
"re_time": 1,
"s_time": 0.1,
"is_logger": False,
"is_test_cdn": True,
"is_cdn": True,
"is_json": False,
},
"loginInitCdn1": { # 登录页面
"req_url": "/otn/resources/login.html",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/view/index.html",
"Host": "kyfw.12306.cn",
"re_try": 1,
"re_time": 1,
"s_time": 0.1,
"is_logger": False,
"is_test_cdn": False,
"is_cdn": True,
"is_json": False,
},
"getDevicesId": { # 获取用户信息
"req_url": "/otn/HttpZF/logdevice",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/passport?redirect=/otn/",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 1,
"s_time": 0.01,
"is_cdn": True,
"is_logger": True,
"is_json": False,
},
"getUserInfo": { # 获取用户信息
"req_url": "/otn/index/initMy12306",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/passport?redirect=/otn/login/userLogin",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 1,
"s_time": 0.01,
"is_cdn": True,
"is_logger": False,
"is_json": False,
},
"userLogin": { # 用户登录
"req_url": "/otn/login/userLogin",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/passport?redirect=/otn/login/userLogin",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"uamauthclient": { # 登录
"req_url": "/otn/uamauthclient",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/passport?redirect=/otn/login/userLogin",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 1,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"initdc_url": { # 生成订单页面
"req_url": "/otn/confirmPassenger/initDc",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.1,
"s_time": 1,
"is_logger": False,
"is_cdn": True,
"is_json": False,
},
"GetJS": { # 订单页面js
"req_url": "/otn/HttpZF/GetJS",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.1,
"s_time": 0.1,
"is_logger": False,
"is_cdn": True,
"is_json": False,
},
"odxmfwg": { # 订单页面js
"req_url": "/otn/dynamicJs/odxmfwg",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.1,
"s_time": 0.1,
"is_logger": False,
"is_cdn": True,
"is_json": False,
},
"get_passengerDTOs": { # 获取乘车人
"req_url": "/otn/confirmPassenger/getPassengerDTOs",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.1,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"select_url": { # 查询余票
"req_url": "/otn/{3}?leftTicketDTO.train_date={0}&leftTicketDTO.from_station={1}&leftTicketDTO.to_station={2}&purpose_codes=ADULT",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 1,
"re_time": 0.01,
"s_time": 0.01,
"is_logger": False,
"is_json": True,
"is_cdn": True,
},
"check_user_url": { # 检查用户登录
"req_url": "/otn/login/checkUser",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 1,
"re_time": 1,
"s_time": 1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"submit_station_url": { # 提交订单
"req_url": "/otn/leftTicket/submitOrderRequest",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"checkOrderInfoUrl": { # 检查订单信息规范
"req_url": "/otn/confirmPassenger/checkOrderInfo",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"getQueueCountUrl": { # 剩余余票数
"req_url": "/otn/confirmPassenger/getQueueCount",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"checkQueueOrderUrl": { # 订单队列排队
"req_url": "/otn/confirmPassenger/confirmSingleForQueue",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"checkRandCodeAnsyn": { # 暂时没用到
"req_url": "/otn/passcodeNew/checkRandCodeAnsyn",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"codeImgByOrder": { # 订单页面验证码
"req_url": "/otn/passcodeNew/getPassCodeNew?module=passenger&rand=randp&{}",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": False,
"is_cdn": True,
"is_json": False,
},
"queryOrderWaitTimeUrl": { # 订单等待页面
"req_url": "/otn/confirmPassenger/queryOrderWaitTime?random={0}&tourFlag=dc&_json_att=",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/confirmPassenger/initDc",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"queryMyOrderNoCompleteUrl": { # 订单查询页面
"req_url": "/otn/queryOrder/queryMyOrderNoComplete",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/queryOrder/initNoComplete",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"initNoCompleteUrl": { # 获取订单列表
"req_url": "/otn/queryOrder/initNoComplete",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/queryOrder/initNoComplete",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": False,
"is_cdn": True,
"is_json": False,
},
"cancelNoCompleteMyOrder": { # 取消订单
"req_url": "/otn/queryOrder/cancelNoCompleteMyOrder",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/queryOrder/initNoComplete",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"autoSubmitOrderRequest": { # 快速自动提交订单
"req_url": "/otn/confirmPassenger/autoSubmitOrderRequest",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"getQueueCountAsync": { # 快速获取订单数据
"req_url": "/otn/confirmPassenger/getQueueCountAsync",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"Content-Type": 1,
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"confirmSingleForQueueAsys": { # 快速订单排队
"req_url": "/otn/confirmPassenger/confirmSingleForQueueAsys",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Content-Type": 1,
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_cdn": True,
"is_json": True,
},
"Pushbear": { # push通知
"req_url": "/sub",
"req_type": "post",
"Referer": "",
"Content-Type": 1,
"Host": "pushbear.ftqq.com",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": False,
"is_json": True,
},
"ServerChan": { # Server酱 push通知
"req_url": "/",
"req_type": "get",
"Referer": "",
"Content-Type": 1,
"Host": "sc.ftqq.com",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.1,
"is_logger": True,
"is_json": True,
},
"loginHtml": { # 登录接口2
"req_url": "/otn/resources/login.html",
"req_type": "get",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.3,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"loginConf": { # 登录接口2
"req_url": "/otn/login/conf",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.3,
"s_time": 0.1,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"loginAysnSuggest": { # 登录接口2
"req_url": "/otn/login/loginAysnSuggest",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.3,
"is_cdn": True,
"s_time": 0.1,
"is_logger": True,
"is_json": True,
},
# 候补订单接口
"chechFace": { # 人脸识别
"req_url": "/otn/afterNate/chechFace",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.01,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"getSuccessRate": { # 成功信息
"req_url": "/otn/afterNate/getSuccessRate",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.01,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"SubmitOrderRequestRsp": { # 提交候补订单准备
"req_url": "/otn/afterNate/submitOrderRequest",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.01,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"confirmHB": { # 设置订单信息
"req_url": "/otn/afterNate/confirmHB",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.01,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"queryQueue": { # 排队
"req_url": "/otn/afterNate/queryQueue",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.01,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"passengerInitApi": { # 排队
"req_url": "/otn/afterNate/passengerInitApi",
"req_type": "post",
"Referer": "https://kyfw.12306.cn/otn/leftTicket/init",
"Host": "kyfw.12306.cn",
"re_try": 10,
"re_time": 0.01,
"s_time": 0.01,
"is_cdn": True,
"is_logger": True,
"is_json": True,
},
"autoVerifyImage": { # 云打码接口
"req_url": TickerConfig.REQ_URL,
"req_type": "post",
"Referer": "",
"Host": TickerConfig.HOST,
"re_try": 6,
"re_time": 10,
"s_time": 0.001,
"is_logger": True,
"is_json": True,
"httpType": TickerConfig.HTTP_TYPE
},
}
================================================
FILE: docker-compose.yml
================================================
version: "3"
services:
#抢票服务
ticket:
build:
context: .
dockerfile: ./Dockerfile37
image: ticket:v1.2.004
environment:
- PYTHONUNBUFFERED=1
- CAPTCHALOCAL=1
container_name: ticket
depends_on:
- captcha
networks:
- 12306network
restart: on-failure
#打码服务器
captcha:
image: yinaoxiong/12306_code_server:amd64 #请根据需要修改image
environment:
- WORKERS=1 #gunicorn works 默认为1可以根据服务器配置自行调整
- PYTHONUNBUFFERED=1
container_name: captcha
networks:
- 12306network
restart: unless-stopped
networks:
12306network:
driver: bridge
================================================
FILE: docker_install_centos.sh
================================================
#!/bin/bash
#author: MonsterTan
#date: 2019-01-15
#this is a script that can install automatically docker software by centos7
function checkSudo (){
if [ $UID -ne 0 ];then
echo -e 'it must be root!'
exit 1
fi
}
checkSudo
## something required system utils
yum install -y yum-utils device-mapper-persistent-data lvm2
## add repo source info
sudo yum-config-manager --add-repo http://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo
sudo yum mackecache fast
sudo yum -y install docker-ce
sudo systemctl start docker
str=successed!
if [ $? -eq 0 ];then
echo -e "\033[32msuccessed!\033[0m"
else
echo -e "\033[31msomething wrong, please check!\033[0m"
fi
echo -e "\033[31mstart to install docker-compose\033[0m"
result=`ls /usr/bin/ | grep ^pip$`
if [ ${#result} -eq 0 ];then
echo -e "\033[31mpip must be necessary, it should be installed firstly\033[0m"
fi
sudo pip install docker-compose
================================================
FILE: filter_cdn_list
================================================
111.161.122.240
112.90.135.96
60.9.0.19
61.162.100.102
222.186.141.146
221.235.187.129
58.221.78.231
113.16.212.251
112.47.27.131
112.123.33.18
183.134.42.18
116.77.75.133
112.90.135.97
58.221.78.42
61.162.100.103
111.161.122.67
60.9.0.20
113.16.212.48
112.47.56.174
112.28.196.75
183.134.42.19
116.77.75.137
112.90.135.98
113.16.212.49
36.250.248.220
60.9.0.21
116.199.127.56
221.235.187.130
112.47.27.172
112.90.135.99
183.134.42.190
60.9.0.22
222.186.141.165
116.77.75.138
36.102.230.136
112.47.56.117
183.134.42.191
60.9.0.23
116.77.75.145
36.102.230.137
222.186.141.166
58.221.78.46
61.162.100.107
116.77.75.144
113.207.77.252
60.9.0.252
222.186.141.178
183.134.42.20
61.167.54.55
58.221.78.47
221.180.208.54
116.77.75.169
61.162.100.252
222.186.141.186
60.9.0.254
113.207.79.17
183.134.42.247
61.167.54.57
111.161.22.17
116.77.75.146
58.221.78.48
61.162.100.41
111.6.176.208
113.207.79.34
36.250.248.222
36.102.230.19
183.134.53.153
116.77.75.147
61.162.100.44
111.6.176.209
113.207.79.42
183.134.53.155
116.77.75.170
36.102.230.20
120.221.64.161
222.218.87.252
111.6.176.248
113.207.79.37
120.221.64.169
112.47.56.118
111.6.176.25
113.194.59.80
124.236.28.100
113.207.81.126
111.6.176.94
111.6.176.95
113.194.59.199
221.235.187.220
124.236.28.252
221.235.187.244
111.6.176.97
124.236.28.67
113.194.59.81
120.221.64.53
36.250.248.252
120.221.64.54
124.236.28.247
36.250.248.254
120.221.64.55
124.236.28.69
116.77.75.183
36.250.248.27
221.235.187.66
183.134.53.223
124.236.28.68
113.207.81.78
58.20.179.253
27.195.145.121
221.235.187.90
113.207.81.81
27.195.145.123
221.235.187.98
123.128.14.254
183.134.53.248
111.161.22.62
123.128.14.69
124.236.28.93
117.169.93.249
183.146.22.135
125.39.1.134
36.102.230.254
124.236.28.92
117.169.93.85
27.195.145.249
125.39.1.191
124.236.28.94
42.81.144.179
222.218.87.28
27.195.145.52
124.236.28.95
36.250.248.56
222.218.87.29
42.81.144.180
27.195.145.62
36.159.115.250
36.159.115.88
36.159.115.89
183.146.22.137
110.242.21.24
124.236.28.98
117.169.93.86
123.128.14.70
110.242.21.23
124.236.28.99
117.23.2.252
123.128.14.72
112.28.196.54
110.242.21.254
117.149.154.185
58.20.179.74
150.138.111.251
117.23.2.28
110.242.21.243
117.149.154.186
58.20.179.73
150.138.111.31
110.242.21.41
117.23.2.29
111.47.220.251
150.138.111.32
110.242.21.71
106.41.0.37
111.47.220.66
110.242.21.70
150.138.111.33
111.47.220.67
106.41.0.44
150.138.167.234
150.138.167.50
106.41.0.45
222.186.145.51
150.138.167.52
111.62.194.30
106.41.0.46
122.228.237.248
42.81.144.31
150.138.169.120
222.186.145.54
59.56.30.51
106.41.0.47
42.81.144.39
111.62.194.31
222.44.151.24
150.138.169.121
112.28.196.251
106.41.0.61
111.62.194.254
123.138.157.85
106.120.178.19
150.138.167.51
150.138.169.123
112.28.196.249
106.41.0.62
120.253.100.20
114.112.172.58
123.138.157.122
150.138.169.124
183.146.22.145
112.28.196.53
120.253.100.21
222.44.151.191
183.146.22.146
125.77.147.254
112.28.196.74
120.253.100.22
222.44.151.25
222.186.145.52
120.253.100.23
61.151.238.197
120.253.100.30
61.151.238.222
61.151.238.229
117.149.155.148
117.149.155.147
183.146.22.171
150.138.169.238
60.210.23.26
125.77.147.68
221.235.187.119
125.77.147.69
115.157.63.19
117.149.155.76
115.157.63.49
122.224.186.221
183.201.225.249
119.52.120.138
60.28.100.155
125.77.147.80
122.70.142.148
122.224.186.222
119.52.120.139
60.28.100.156
122.70.142.147
125.77.147.82
122.224.186.225
60.28.100.157
119.52.120.140
125.77.147.88
114.112.172.57
122.224.186.226
60.28.100.158
150.138.214.84
119.52.120.144
58.216.21.250
60.28.100.248
119.52.120.145
150.138.214.124
60.210.23.116
183.66.109.254
117.180.229.254
150.138.214.85
119.52.120.146
61.147.226.46
115.157.63.50
122.224.186.253
58.216.22.17
150.138.214.86
222.186.145.53
117.180.229.142
58.216.22.20
183.222.97.164
61.147.226.48
58.216.22.22
183.222.97.165
122.224.186.223
183.66.109.44
58.216.22.56
183.222.97.166
60.210.23.29
221.235.187.67
183.66.109.45
115.157.63.51
183.222.97.254
183.146.22.142
183.146.22.139
120.226.55.144
221.235.187.121
153.99.235.112
120.226.55.151
120.226.55.254
112.240.60.213
36.25.241.75
122.225.83.25
153.99.235.91
112.240.60.214
122.225.83.28
115.157.63.52
36.25.241.76
121.22.247.202
114.112.172.59
122.228.237.74
36.25.241.77
112.240.60.215
121.22.247.204
223.111.18.161
122.228.237.75
112.240.60.216
121.22.247.254
218.26.75.149
223.111.18.162
122.228.237.76
112.240.60.217
115.157.63.62
218.26.75.150
223.111.18.163
122.225.83.26
122.228.239.233
112.240.60.222
223.111.18.217
218.26.75.151
222.186.145.251
122.228.239.234
112.240.60.236
218.26.75.152
116.207.132.183
122.228.239.235
61.147.227.53
159.226.225.149
218.26.75.153
116.207.132.184
122.228.239.236
116.207.132.253
218.26.75.206
113.207.10.204
122.228.239.238
218.26.75.236
113.207.10.218
122.228.239.243
122.228.239.244
122.228.239.246
139.209.49.140
183.131.124.38
159.226.225.154
139.209.49.138
121.22.247.203
139.209.49.144
123.53.139.253
139.209.49.151
125.74.58.134
139.209.49.152
123.53.139.36
125.74.58.135
125.74.58.136
139.209.49.153
123.184.108.251
210.38.3.23
183.131.124.40
123.184.108.60
123.184.108.61
210.38.3.24
159.226.225.139
183.131.124.58
125.74.58.254
210.38.3.42
159.226.225.140
61.54.7.158
124.95.148.254
183.131.124.59
61.54.7.174
210.38.3.49
124.95.148.30
101.69.146.234
210.38.3.50
61.54.7.179
124.95.148.29
14.204.185.100
210.38.3.60
61.54.7.243
60.220.196.220
124.95.148.31
14.204.185.101
60.220.196.221
59.83.232.18
14.204.185.102
59.83.232.50
61.147.210.193
122.70.142.252
14.204.185.123
61.147.210.195
14.204.185.254
223.111.198.252
61.147.226.185
101.69.146.33
223.111.198.65
113.207.70.194
61.147.210.242
113.207.70.207
101.69.146.35
60.210.23.23
61.147.226.47
113.207.70.208
101.69.146.34
60.210.23.25
61.147.226.49
183.131.168.148
60.210.23.27
61.147.227.102
113.207.72.102
60.213.21.117
58.220.71.253
61.147.227.126
58.220.71.51
60.213.21.118
14.204.185.91
58.220.71.52
113.207.70.209
60.213.21.156
113.207.77.117
60.213.21.214
60.220.196.250
58.220.71.62
116.199.127.50
60.213.21.244
58.220.71.63
118.123.233.254
116.199.127.54
218.29.198.43
58.220.71.64
60.213.21.243
118.123.233.30
218.29.198.62
116.77.73.164
103.254.189.230
58.220.71.65
60.213.21.245
118.123.233.31
223.86.219.254
183.214.132.120
60.213.21.252
223.86.219.65
183.214.132.16
116.199.127.55
223.86.219.66
183.214.132.17
60.255.143.93
61.149.9.150
60.255.143.94
60.255.143.95
60.255.143.96
36.250.233.185
106.40.140.224
36.250.233.208
61.147.228.201
36.250.233.209
183.214.140.204
123.53.139.37
222.24.122.86
61.147.228.60
36.250.233.210
106.40.140.254
183.214.140.238
36.250.233.214
36.250.233.228
58.221.78.186
36.250.248.218
218.12.228.246
112.90.135.244
222.24.122.89
36.250.248.217
221.235.187.106
218.12.228.38
36.250.233.254
112.47.20.250
218.12.228.39
112.90.135.92
117.27.245.227
112.90.135.93
117.27.245.254
117.27.245.52
112.90.135.94
117.27.245.97
112.90.135.95
117.27.245.54
218.60.185.251
113.5.80.33
125.77.130.247
218.60.185.46
125.77.130.251
118.203.202.206
118.203.202.207
218.60.185.47
218.60.185.48
118.203.202.208
118.203.202.209
118.203.202.222
221.180.192.100
125.77.130.46
125.77.130.47
112.47.20.68
125.77.130.48
163.177.132.27
112.47.27.132
125.77.130.49
112.47.20.89
61.147.228.89
112.47.20.79
106.120.178.253
221.180.192.254
221.230.141.170
221.180.192.61
106.120.178.20
163.177.132.30
183.216.176.74
221.230.141.172
183.216.176.75
221.230.141.174
110.19.204.219
110.19.204.220
219.138.27.108
110.19.204.221
219.138.27.249
103.254.189.229
106.120.178.22
219.138.27.30
219.138.27.31
113.16.208.251
222.186.141.132
113.16.208.77
113.16.208.78
222.186.141.141
110.19.204.254
222.186.141.142
221.180.208.252
222.186.141.143
117.161.19.126
221.180.208.46
222.186.141.145
60.9.0.18
117.161.19.19
221.180.208.47
117.161.19.22
113.142.80.223
183.134.42.153
183.134.42.154
183.134.42.155
115.223.24.254
183.134.42.156
111.161.122.100
183.134.42.157
61.136.167.17
183.134.42.158
61.136.167.18
183.134.42.159
113.142.80.69
183.134.42.17
113.142.80.71
61.136.167.19
113.142.80.72
115.223.24.80
111.161.122.133
115.223.24.81
111.161.122.134
115.223.24.82
111.161.122.135
114.112.172.56
222.74.113.196
222.74.113.219
61.167.54.236
61.167.54.242
================================================
FILE: init/__init__.py
================================================
================================================
FILE: init/login.py
================================================
# -*- coding=utf-8 -*-
import copy
import time
from collections import OrderedDict
from time import sleep
import TickerConfig
from inter.GetPassCodeNewOrderAndLogin import getPassCodeNewOrderAndLogin1
from inter.GetRandCode import getRandCode
from inter.LoginAysnSuggest import loginAysnSuggest
from inter.LoginConf import loginConf
from myException.UserPasswordException import UserPasswordException
class GoLogin:
def __init__(self, session, is_auto_code, auto_code_type):
self.session = session
self.randCode = ""
self.is_auto_code = is_auto_code
self.auto_code_type = auto_code_type
def auth(self):
"""
:return:
"""
self.session.httpClint.send(self.session.urls["loginInitCdn1"])
uamtkStaticUrl = self.session.urls["uamtk-static"]
uamtkStaticData = {"appid": "otn"}
return self.session.httpClint.send(uamtkStaticUrl, uamtkStaticData)
def codeCheck(self):
"""
验证码校验
:return:
"""
codeCheckUrl = copy.deepcopy(self.session.urls["codeCheck1"])
codeCheckUrl["req_url"] = codeCheckUrl["req_url"].format(self.randCode, int(time.time() * 1000))
fresult = self.session.httpClint.send(codeCheckUrl)
if not isinstance(fresult, str):
print("登录失败")
return
fresult = eval(fresult.split("(")[1].split(")")[0])
if "result_code" in fresult and fresult["result_code"] == "4":
print(u"验证码通过,开始登录..")
return True
else:
if "result_message" in fresult:
print(fresult["result_message"])
sleep(1)
self.session.httpClint.del_cookies()
def baseLogin(self, user, passwd):
"""
登录过程
:param user:
:param passwd:
:return: 权限校验码
"""
logurl = self.session.urls["login"]
loginData = OrderedDict()
loginData["username"] = user,
loginData["password"] = passwd,
loginData["appid"] = "otn",
loginData["answer"] = self.randCode,
tresult = self.session.httpClint.send(logurl, loginData)
if 'result_code' in tresult and tresult["result_code"] == 0:
print(u"登录成功")
tk = self.auth()
if "newapptk" in tk and tk["newapptk"]:
return tk["newapptk"]
else:
return False
elif 'result_message' in tresult and tresult['result_message']:
messages = tresult['result_message']
if messages.find(u"密码输入错误") is not -1:
raise UserPasswordException("{0}".format(messages))
else:
print(u"登录失败: {0}".format(messages))
print(u"尝试重新登陆")
return False
else:
return False
def getUserName(self, uamtk):
"""
登录成功后,显示用户名
:return:
"""
if not uamtk:
return u"权限校验码不能为空"
else:
uamauthclientUrl = self.session.urls["uamauthclient"]
data = {"tk": uamtk}
uamauthclientResult = self.session.httpClint.send(uamauthclientUrl, data)
if uamauthclientResult:
if "result_code" in uamauthclientResult and uamauthclientResult["result_code"] == 0:
print(u"欢迎 {} 登录".format(uamauthclientResult["username"]))
return True
else:
return False
else:
self.session.httpClint.send(uamauthclientUrl, data)
url = self.session.urls["getUserInfo"]
self.session.httpClint.send(url)
def go_login(self):
"""
登陆
:param user: 账户名
:param passwd: 密码
:return:
"""
user, passwd = TickerConfig.USER, TickerConfig.PWD
if not user or not passwd:
raise UserPasswordException(u"温馨提示: 用户名或者密码为空,请仔细检查")
login_num = 0
while True:
if loginConf(self.session):
result = getPassCodeNewOrderAndLogin1(session=self.session, imgType="login")
if not result:
continue
self.randCode = getRandCode(self.is_auto_code, self.auto_code_type, result)
print(self.randCode)
login_num += 1
self.auth()
if self.codeCheck():
uamtk = self.baseLogin(user, passwd)
if uamtk:
self.getUserName(uamtk)
break
else:
loginAysnSuggest(self.session, username=user, password=passwd)
login_num += 1
break
================================================
FILE: init/select_ticket_info.py
================================================
# -*- coding=utf-8 -*-
import datetime
import random
import os
import socket
import sys
import threading
import time
import TickerConfig
import wrapcache
from agency.cdn_utils import CDNProxy, open_cdn_file
from config import urlConf, configCommon
from config.TicketEnmu import ticket
from config.configCommon import seat_conf_2, seat_conf
from config.getCookie import getDrvicesID
from init.login import GoLogin
from inter.AutoSubmitOrderRequest import autoSubmitOrderRequest
from inter.ChechFace import chechFace
from inter.CheckUser import checkUser
from inter.GetPassengerDTOs import getPassengerDTOs
from inter.LiftTicketInit import liftTicketInit
from inter.Query import query
from inter.SubmitOrderRequest import submitOrderRequest
from myException.PassengerUserException import PassengerUserException
from myException.UserPasswordException import UserPasswordException
from myException.ticketConfigException import ticketConfigException
from myException.ticketIsExitsException import ticketIsExitsException
from myException.ticketNumOutException import ticketNumOutException
from myUrllib.httpUtils import HTTPClient
class select:
"""
快速提交车票通道
"""
def __init__(self):
self.cdn_list = open_cdn_file("filter_cdn_list")
self.get_ticket_info()
self._station_seat = [seat_conf[x] for x in TickerConfig.SET_TYPE]
self.auto_code_type = TickerConfig.AUTO_CODE_TYPE
self.httpClint = HTTPClient(TickerConfig.IS_PROXY, self.cdn_list)
self.httpClint.cdn = self.cdn_list[random.randint(0, 4)]
self.urls = urlConf.urls
self.login = GoLogin(self, TickerConfig.IS_AUTO_CODE, self.auto_code_type)
self.cookies = ""
self.queryUrl = "leftTicket/queryO"
self.passengerTicketStrList = ""
self.passengerTicketStrByAfterLate = ""
self.oldPassengerStr = ""
self.set_type = ""
self.flag = True
@staticmethod
def get_ticket_info():
"""
获取配置信息
:return:
"""
print(u"*" * 50)
print(f"检查当前版本为: {TickerConfig.RE_VERSION}")
version = sys.version.split(" ")[0]
print(u"检查当前python版本为:{},目前版本只支持3.6以上".format(version))
if version < "3.6.0":
raise Exception
print(u"12306刷票小助手,最后更新于2019.09.18,请勿作为商业用途,交流群号:"
u" 1群:286271084(已满)\n"
u" 2群:649992274(已满)\n"
u" 3群:632501142(已满)\n"
u" 4群: 606340519(已满)\n"
u" 5群: 948526733(已满)\n"
u" 7群: 660689659(已满)\n"
u" 8群: 620629239(已满)\n"
u" 6群: 608792930(未满)\n"
u" 9群: 693035807(未满)\n"
)
print(
f"当前配置:\n出发站:{TickerConfig.FROM_STATION}\n到达站:{TickerConfig.TO_STATION}\n车次: {','.join(TickerConfig.STATION_TRAINS) or '所有车次'}\n乘车日期:{','.join(TickerConfig.STATION_DATES)}\n坐席:{','.join(TickerConfig.SET_TYPE)}\n是否有票优先提交:{TickerConfig.IS_MORE_TICKET}\n乘车人:{TickerConfig.TICKET_PEOPLES}\n" \
f"刷新间隔: 随机(1-3S)\n僵尸票关小黑屋时长: {TickerConfig.TICKET_BLACK_LIST_TIME}\n下单接口: {TickerConfig.ORDER_TYPE}\n下单模式: {TickerConfig.ORDER_MODEL}\n预售踩点时间:{TickerConfig.OPEN_TIME}")
print(u"*" * 50)
def station_table(self, from_station, to_station):
"""
读取车站信息
:param station:
:return:
"""
path = os.path.join(os.path.dirname(__file__), '../station_name.txt')
try:
with open(path, encoding="utf-8") as result:
info = result.read().split('=')[1].strip("'").split('@')
except Exception:
with open(path) as result:
info = result.read().split('=')[1].strip("'").split('@')
del info[0]
station_name = {}
for i in range(0, len(info)):
n_info = info[i].split('|')
station_name[n_info[1]] = n_info[2]
try:
from_station = station_name[from_station.encode("utf8")]
to_station = station_name[to_station.encode("utf8")]
except KeyError:
from_station = station_name[from_station]
to_station = station_name[to_station]
return from_station, to_station
def call_login(self, auth=False):
"""
登录回调方法
:return:
"""
if auth:
return self.login.auth()
else:
configCommon.checkSleepTime(self) # 防止网上启动晚上到点休眠
self.login.go_login()
def main(self):
l = liftTicketInit(self)
l.reqLiftTicketInit()
getDrvicesID(self)
self.call_login()
check_user = checkUser(self)
t = threading.Thread(target=check_user.sendCheckUser)
t.setDaemon(True)
t.start()
from_station, to_station = self.station_table(TickerConfig.FROM_STATION, TickerConfig.TO_STATION)
num = 0
s = getPassengerDTOs(selectObj=self, ticket_peoples=TickerConfig.TICKET_PEOPLES)
passenger = s.sendGetPassengerDTOs()
wrapcache.set("user_info", passenger, timeout=9999999)
now = datetime.datetime.now()
if TickerConfig.ORDER_MODEL is 1:
print(f"预售还未开始,阻塞中,预售时间为{TickerConfig.OPEN_TIME}, 当前时间为: {now.strftime('%H:%M:%S')}")
sleep_time_s = 0.1
sleep_time_t = 0.3
# 测试了一下有微妙级的误差,应该不影响,测试结果:2019-01-02 22:30:00.004555,预售还是会受到前一次刷新的时间影响,暂时没想到好的解决方案
while now.strftime("%H:%M:%S") < TickerConfig.OPEN_TIME:
now = datetime.datetime.now()
time.sleep(0.0001)
print(f"预售开始,开启时间为: {now.strftime('%H:%M:%S')}")
else:
sleep_time_s = TickerConfig.MIN_TIME
sleep_time_t = TickerConfig.MAX_TIME
while 1:
try:
num += 1
now = datetime.datetime.now() # 感谢群里大佬提供整点代码
configCommon.checkSleepTime(self) # 晚上到点休眠
q = query(selectObj=self,
from_station=from_station,
to_station=to_station,
from_station_h=TickerConfig.FROM_STATION,
to_station_h=TickerConfig.TO_STATION,
_station_seat=self._station_seat,
station_trains=TickerConfig.STATION_TRAINS,
station_dates=TickerConfig.STATION_DATES,
ticke_peoples_num=len(TickerConfig.TICKET_PEOPLES),
)
queryResult = q.sendQuery()
# 查询接口
if queryResult.get("status"):
train_no = queryResult.get("train_no", "")
train_date = queryResult.get("train_date", "")
stationTrainCode = queryResult.get("stationTrainCode", "")
secretStr = queryResult.get("secretStr", "")
secretList = queryResult.get("secretList", "")
seat = queryResult.get("seat", "")
leftTicket = queryResult.get("leftTicket", "")
query_from_station_name = queryResult.get("query_from_station_name", "")
query_to_station_name = queryResult.get("query_to_station_name", "")
is_more_ticket_num = queryResult.get("is_more_ticket_num", len(TickerConfig.TICKET_PEOPLES))
if wrapcache.get(train_no):
print(ticket.QUEUE_WARNING_MSG.format(train_no))
else:
# 获取联系人
s = getPassengerDTOs(selectObj=self, ticket_peoples=TickerConfig.TICKET_PEOPLES,
set_type="" if isinstance(seat, list) else seat_conf_2[seat],
# 候补订单需要设置多个坐席
is_more_ticket_num=is_more_ticket_num)
getPassengerDTOsResult = s.getPassengerTicketStrListAndOldPassengerStr(secretStr, secretList)
if getPassengerDTOsResult.get("status", False):
self.passengerTicketStrList = getPassengerDTOsResult.get("passengerTicketStrList", "")
self.passengerTicketStrByAfterLate = getPassengerDTOsResult.get(
"passengerTicketStrByAfterLate", "")
self.oldPassengerStr = getPassengerDTOsResult.get("oldPassengerStr", "")
self.set_type = getPassengerDTOsResult.get("set_type", "")
# 提交订单
# 订单分为两种,一种为抢单,一种为候补订单
if secretStr: # 正常下单
if TickerConfig.ORDER_TYPE == 1: # 快速下单
a = autoSubmitOrderRequest(selectObj=self,
secretStr=secretStr,
train_date=train_date,
passengerTicketStr=self.passengerTicketStrList,
oldPassengerStr=self.oldPassengerStr,
train_no=train_no,
stationTrainCode=stationTrainCode,
leftTicket=leftTicket,
set_type=self.set_type,
query_from_station_name=query_from_station_name,
query_to_station_name=query_to_station_name,
)
a.sendAutoSubmitOrderRequest()
elif TickerConfig.ORDER_TYPE == 2: # 普通下单
sor = submitOrderRequest(self, secretStr, from_station, to_station, train_no,
self.set_type,
self.passengerTicketStrList, self.oldPassengerStr, train_date,
TickerConfig.TICKET_PEOPLES)
sor.sendSubmitOrderRequest()
elif secretList: # 候补订单
c = chechFace(self, secretList, train_no)
c.sendChechFace()
else:
random_time = round(random.uniform(sleep_time_s, sleep_time_t), 2)
nateMsg = ' 无候补机会' if TickerConfig.ORDER_TYPE == 2 else ""
print(f"正在第{num}次查询 停留时间:{random_time} 乘车日期: {','.join(TickerConfig.STATION_DATES)} 车次:{','.join(TickerConfig.STATION_TRAINS) or '所有车次'} 下单无票{nateMsg} 耗时:{(datetime.datetime.now() - now).microseconds / 1000} {queryResult.get('cdn')}")
time.sleep(random_time)
except PassengerUserException as e:
print(e)
break
except ticketConfigException as e:
print(e)
break
except ticketIsExitsException as e:
print(e)
break
except ticketNumOutException as e:
print(e)
break
except UserPasswordException as e:
print(e)
break
except ValueError as e:
if e == "No JSON object could be decoded":
print(u"12306接口无响应,正在重试")
else:
print(e)
except KeyError as e:
print(e)
except TypeError as e:
print(u"12306接口无响应,正在重试 {0}".format(e))
except socket.error as e:
print(e)
if __name__ == '__main__':
s = select()
cdn = s.station_table("长沙", "深圳")
================================================
FILE: inter/AutoSubmitOrderRequest.py
================================================
# coding=utf-8
import urllib
from collections import OrderedDict
from config.TicketEnmu import ticket
from inter.CheckRandCodeAnsyn import checkRandCodeAnsyn
from inter.GetQueueCountAsync import getQueueCountAsync
from inter.GetRandCode import getRandCode
import TickerConfig
class autoSubmitOrderRequest:
"""
快读提交订单通道
"""
def __init__(self, selectObj,
secretStr,
train_date,
query_from_station_name,
query_to_station_name,
passengerTicketStr,
oldPassengerStr,
train_no,
stationTrainCode,
leftTicket,
set_type,):
self.set_type = set_type
try:
self.secretStr = urllib.unquote(secretStr)
except AttributeError:
self.secretStr = urllib.parse.unquote(secretStr)
self.train_date = train_date
self.query_from_station_name = query_from_station_name
self.query_to_station_name = query_to_station_name
self.passengerTicketStr = passengerTicketStr.rstrip("_{0}".format(self.set_type))
self.oldPassengerStr = oldPassengerStr
self.session = selectObj
self.train_no = train_no
self.stationTrainCode = stationTrainCode
self.leftTicket = leftTicket
def data_par(self):
"""
参数结构
自动提交代码接口-autoSubmitOrderRequest
- 字段说明
- secretStr 车票代码
- train_date 乘车日期
- tour_flag 乘车类型
- purpose_codes 学生还是成人
- query_from_station_name 起始车站
- query_to_station_name 结束车站
- cancel_flag 默认2,我也不知道干嘛的
- bed_level_order_num 000000000000000000000000000000
- passengerTicketStr 乘客乘车代码
- oldPassengerStr 乘客编号代码
:return:
"""
data = OrderedDict()
data["secretStr"] = self.secretStr
data["train_date"] = self.train_date
data["tour_flag"] = "dc"
data["purpose_codes"] = "ADULT"
data["query_from_station_name"] = TickerConfig.FROM_STATION
data["query_to_station_name"] = TickerConfig.TO_STATION
data["cancel_flag"] = 2
data["bed_level_order_num"] = "000000000000000000000000000000"
data["passengerTicketStr"] = self.passengerTicketStr
data["oldPassengerStr"] = self.oldPassengerStr
return data
def sendAutoSubmitOrderRequest(self):
"""
请求下单接口
:return:
"""
urls = self.session.urls["autoSubmitOrderRequest"]
data = self.data_par()
autoSubmitOrderRequestResult = self.session.httpClint.send(urls, data)
if autoSubmitOrderRequestResult and \
autoSubmitOrderRequestResult.get("status", False) and\
autoSubmitOrderRequestResult.get("httpstatus", False) == 200:
requestResultData = autoSubmitOrderRequestResult.get("data", {})
if requestResultData:
result = requestResultData.get("result", "")
ifShowPassCode = requestResultData.get("ifShowPassCode", "N")
ifShowPassCodeTime = int(requestResultData.get("ifShowPassCodeTime", "1000")) / float(1000)
print(ticket.AUTO_SUBMIT_ORDER_REQUEST_C)
g = getQueueCountAsync(session=self.session,
train_no=self.train_no,
stationTrainCode=self.stationTrainCode,
fromStationTelecode=self.query_from_station_name,
toStationTelecode=self.query_to_station_name,
leftTicket=self.leftTicket,
set_type=self.set_type,
users=len(TickerConfig.TICKET_PEOPLES),
station_dates=self.train_date,
passengerTicketStr=self.passengerTicketStr,
oldPassengerStr=self.oldPassengerStr,
result=result,
ifShowPassCodeTime=ifShowPassCodeTime,
)
if ifShowPassCode == "Y": # 如果需要验证码
print(u"需要验证码")
print(u"正在使用自动识别验证码功能")
for i in range(3):
randCode = getRandCode(is_auto_code=True, auto_code_type=2)
checkcode = checkRandCodeAnsyn(self.session, randCode, "")
if checkcode == 'TRUE':
print(u"验证码通过,正在提交订单")
data['randCode'] = randCode
break
else:
print (u"验证码有误, {0}次尝试重试".format(i + 1))
print(u"验证码超过限定次数3次,放弃此次订票机会!")
g.sendGetQueueCountAsync()
else:
print(ticket.AUTO_SUBMIT_ORDER_REQUEST_F)
if autoSubmitOrderRequestResult.get("messages", ""):
print("".join(autoSubmitOrderRequestResult.get("messages", "")))
elif autoSubmitOrderRequestResult.get("validateMessages", ""):
print("".join(autoSubmitOrderRequestResult.get("validateMessages", "")))
================================================
FILE: inter/ChechFace.py
================================================
import datetime
import urllib
from collections import OrderedDict
from config.urlConf import urls
import TickerConfig
from inter.GetSuccessRate import getSuccessRate
from myException.ticketConfigException import ticketConfigException
import wrapcache
class chechFace:
def __init__(self, selectObj, secretList, train_no):
"""
人脸识别
"""
self.secretList = secretList
self.session = selectObj
self.train_no = train_no
def data_apr(self):
"""
secretList 9vqa9%2B%2F%2Fsdozmm22hpSeDTGqRUwSuA2D0r%2BmU%2BLZj7MK7CDuf5Ep1xpxl4Dyxfmoah%2BaB9TZSesU%0AkxBbo5oNgR1vqMfvq66VP0T7tpQtH%2BbVGBz1FolZG8jDD%2FHqnz%2FnvdBP416Og6WGS14O%2F3iBSwT8%0AkRPsNF0Vq0U082g0tlJtP%2BPn7TzW3z7TDCceMJIjFcfEOA%2BW%2BuK%2Bpy6jCQMv0TmlkXf5aKcGnE02%0APuv4I8nF%2BOWjWzv9CrJyiCZiWaXd%2Bi7p69V3a9dhF787UgS660%2BqKRFB4RLwAfic3MkAlfpGWhMY%0ACfARVQ%3D%3D#O|
_json_att
候补一次只能补一个座位,默认取TICKET_TYPE第一个
:return:
"""
ticker = TickerConfig.PASSENGER_TICKER_STR.get(TickerConfig.SET_TYPE[0])
data = OrderedDict()
data["secretList"] = f"{self.secretList}#{ticker}|"
data["_json_att"] = ""
return data
def sendChechFace(self):
chechFaceRsp = self.session.httpClint.send(urls.get("chechFace"), self.data_apr())
if not chechFaceRsp.get("status"):
print("".join(chechFaceRsp.get("messages")) or chechFaceRsp.get("validateMessages"))
wrapcache.set(key=f"hb{self.train_no}", value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
return
data = chechFaceRsp["data"]
if not data.get("face_flag"):
print("".join(chechFaceRsp.get("messages")) or chechFaceRsp.get("validateMessages"))
if data.get("face_check_code") == "14":
"""
未通过人脸核验
"""
raise ticketConfigException("通过人证一致性核验的用户及激活的“铁路畅行”会员可以提交候补需求,请您按照操作说明在铁路12306app.上完成人证核验")
elif data.get("face_check_code") in ["12", "02"]:
"""
系统忙,请稍后再试!
"""
print("系统忙,请稍后再试!")
wrapcache.set(key=f"hb{self.train_no}", value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
elif data.get("face_check_code") in ["03", "13"]:
"""
证件信息审核失败,请检查所填写的身份信息内容与原证件是否一致。
"""
raise ticketConfigException("证件信息审核失败,请检查所填写的身份信息内容与原证件是否一致。")
elif data.get("face_check_code") in ["01", "11"]:
"""
证件信息正在审核中,请您耐心等待,审核通过后可继续完成候补操作。
"""
print("证件信息正在审核中,请您耐心等待,审核通过后可继续完成候补操作。")
wrapcache.set(key=f"hb{self.train_no}", value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
g = getSuccessRate(self.session, self.secretList)
g.sendSuccessRate()
================================================
FILE: inter/CheckOrderInfo.py
================================================
# coding=utf-8
from collections import OrderedDict
from inter.GetQueueCount import getQueueCount
from inter.GetRepeatSubmitToken import getRepeatSubmitToken
class checkOrderInfo:
def __init__(self, session, train_no, set_type, passengerTicketStrList, oldPassengerStr, station_dates, ticket_peoples):
self.train_no = train_no
self.set_type = set_type
self.passengerTicketStrList = passengerTicketStrList
self.oldPassengerStr = oldPassengerStr
self.station_dates = station_dates
self.ticket_peoples = ticket_peoples
self.RepeatSubmitToken = getRepeatSubmitToken(session)
self.getTicketInfoForPassengerForm = self.RepeatSubmitToken.sendGetRepeatSubmitToken()
self.ticketInfoForPassengerForm = self.getTicketInfoForPassengerForm.get("ticketInfoForPassengerForm", "")
self.token = self.getTicketInfoForPassengerForm.get("token", "")
self.session = self.getTicketInfoForPassengerForm.get("session", "")
def data_par(self):
"""
参数结构
:return:
"""
data = OrderedDict()
data['bed_level_order_num'] = "000000000000000000000000000000"
data['passengerTicketStr'] = self.passengerTicketStrList.rstrip("_{0}".format(self.set_type))
data['oldPassengerStr'] = self.oldPassengerStr
data['tour_flag'] = 'dc'
data['randCode'] = ""
data['cancel_flag'] = 2
data['_json_att'] = ""
data['REPEAT_SUBMIT_TOKEN'] = self.token
return data
def sendCheckOrderInfo(self):
"""
检查支付订单,需要提交REPEAT_SUBMIT_TOKEN
passengerTicketStr : 座位编号,0,票类型,乘客名,证件类型,证件号,手机号码,保存常用联系人(Y或N)
oldPassengersStr: 乘客名,证件类型,证件号,乘客类型
:return:
"""
CheckOrderInfoUrls = self.session.urls["checkOrderInfoUrl"]
data = self.data_par()
checkOrderInfoRep = self.session.httpClint.send(CheckOrderInfoUrls, data)
data = checkOrderInfoRep.get("data", {})
if data and data.get("submitStatus", False):
print (u'车票提交通过,正在尝试排队')
ifShowPassCodeTime = int(checkOrderInfoRep["data"]["ifShowPassCodeTime"]) / float(1000)
if "ifShowPassCode" in checkOrderInfoRep["data"] and checkOrderInfoRep["data"]["ifShowPassCode"] == "Y":
is_need_code = True
elif "ifShowPassCode" in checkOrderInfoRep["data"] and checkOrderInfoRep['data']['submitStatus'] is True:
is_need_code = False
else:
is_need_code = False
QueueCount = getQueueCount(self.session,
is_need_code,
ifShowPassCodeTime,
self.set_type,
self.station_dates,
self.train_no,
self.ticket_peoples,
self.ticketInfoForPassengerForm,
self.token,
self.oldPassengerStr,
self.passengerTicketStrList,
)
QueueCount.sendGetQueueCount()
elif "errMsg" in data and data["errMsg"]:
print(checkOrderInfoRep['data']["errMsg"])
elif 'messages' in checkOrderInfoRep and checkOrderInfoRep['messages']:
print (checkOrderInfoRep['messages'][0])
================================================
FILE: inter/CheckRandCodeAnsyn.py
================================================
# coding=utf-8
class checkRandCodeAnsyn:
def __init__(self, session, randCode, token):
self.session = session
self.randCode = randCode
self.token = token
def data_par(self):
"""
:return:
"""
data = {
"randCode": self.randCode,
"rand": "randp",
"_json_att": "",
"REPEAT_SUBMIT_TOKEN": self.token
}
return data
def sendCheckRandCodeAnsyn(self):
"""
下单验证码识别
:return:
"""
checkRandCodeAnsynUrl = self.session.urls["checkRandCodeAnsyn"]
fresult = self.session.httpClint.send(checkRandCodeAnsynUrl, self.data_par()) # 校验验证码是否正确
return fresult['data']['msg']
================================================
FILE: inter/CheckUser.py
================================================
# coding=utf-8
import datetime
import random
import time
import wrapcache
from config import configCommon
from config.TicketEnmu import ticket
class checkUser:
def __init__(self, session):
self.session = session
def sendCheckUser(self):
"""
检查用户登录, 检查间隔为2分钟
:return:
"""
CHENK_TIME = 1
while 1:
time.sleep(3) # 防止cpu占用过高
configCommon.checkSleepTime(self.session) # 修复晚上查询线程休眠时,检查登录线程为休眠,造成快豆迅速消耗
if wrapcache.get("user_time") is None:
check_user_url = self.session.urls["check_user_url"]
data = {"_json_att": ""}
check_user = self.session.httpClint.send(check_user_url, data)
if check_user.get("data", False):
check_user_flag = check_user["data"]["flag"]
if check_user_flag is True:
wrapcache.set("user_time", datetime.datetime.now(), timeout=random.randint(60, 80) * CHENK_TIME)
else:
if check_user['messages']:
print(ticket.LOGIN_SESSION_FAIL.format(check_user['messages']))
self.session.call_login()
wrapcache.set("user_time", datetime.datetime.now(), timeout=random.randint(60, 80) * CHENK_TIME)
else:
print(ticket.LOGIN_SESSION_FAIL.format(check_user['messages']))
self.session.call_login()
wrapcache.set("user_time", datetime.datetime.now(), timeout=random.randint(60, 80) * CHENK_TIME)
================================================
FILE: inter/ConfirmHB.py
================================================
from collections import OrderedDict
from config.urlConf import urls
import TickerConfig
from inter.GetQueueCount import queryQueueByAfterNate
class confirmHB:
def __init__(self, secretList, session, tickerNo, jzdhDate):
"""
人脸识别
"""
self.secretList = secretList
self.session = session
self.passengerTicketStrByAfterLate = session.passengerTicketStrByAfterLate
self.tickerNo = tickerNo
self.jzdhDate = jzdhDate
def data_apr(self):
"""
passengerInfo 1#XXXX#1#***************77X#bf6ae40d3655ae7eff005ee21d95876b38ab97a8031b464bc2f74a067e3ec957;
jzParam 2019-08-31#19#00
hbTrain 5l000G177230,O#
lkParam
:return:
"""
ticker = TickerConfig.PASSENGER_TICKER_STR.get(TickerConfig.SET_TYPE[0])
data = OrderedDict()
data["passengerInfo"] = self.passengerTicketStrByAfterLate
data["jzParam"] = self.jzdhDate
data["hbTrain"] = f"{self.tickerNo},{ticker}#"
data["lkParam"] = ""
return data
def sendChechFace(self):
ChechFaceRsp = self.session.httpClint.send(urls.get("confirmHB"), self.data_apr())
if not ChechFaceRsp.get("status"):
print("".join(ChechFaceRsp.get("messages")) or ChechFaceRsp.get("validateMessages"))
return
data = ChechFaceRsp.get("data")
if not data.get("flag"):
print(f"错误信息:{data.get('msg')}")
return
queue = queryQueueByAfterNate(self.session)
queue.sendQueryQueueByAfterNate()
================================================
FILE: inter/ConfirmSingleForQueue.py
================================================
# coding=utf-8
import datetime
import time
from inter.CheckRandCodeAnsyn import checkRandCodeAnsyn
from inter.GetPassengerDTOs import getPassengerDTOs
from inter.GetRandCode import getRandCode
from inter.QueryOrderWaitTime import queryOrderWaitTime
class confirmSingleForQueue:
def __init__(self, session, ifShowPassCodeTime, is_node_code, token, set_type, ticket_peoples, ticketInfoForPassengerForm,
oldPassengerStr, passengerTicketStrList):
self.session = session
self.ifShowPassCodeTime = ifShowPassCodeTime
self.is_node_code = is_node_code
self.token = token
self.set_type = set_type
self.ticket_peoples = ticket_peoples
self.ticketInfoForPassengerForm = ticketInfoForPassengerForm
self.passengerTicketStrList = passengerTicketStrList
self.oldPassengerStr = oldPassengerStr
def data_par(self):
"""
模拟提交订单是确认按钮,参数获取方法还是get_ticketInfoForPassengerForm 中获取
:return:
"""
if not self.passengerTicketStrList and not self.oldPassengerStr:
s = getPassengerDTOs(session=self.session, ticket_peoples=self.ticket_peoples, set_type=self.set_type)
getPassengerDTOsResult = s.getPassengerTicketStrListAndOldPassengerStr()
if getPassengerDTOsResult.get("status", False):
self.passengerTicketStrList = getPassengerDTOsResult.get("passengerTicketStrList", "")
self.oldPassengerStr = getPassengerDTOsResult.get("oldPassengerStr", "")
data = {
"passengerTicketStr": self.passengerTicketStrList.rstrip("_{0}".format(self.set_type)),
"oldPassengerStr": "".join(self.oldPassengerStr),
"purpose_codes": self.ticketInfoForPassengerForm["purpose_codes"],
"key_check_isChange": self.ticketInfoForPassengerForm["key_check_isChange"],
"leftTicketStr": self.ticketInfoForPassengerForm["leftTicketStr"],
"train_location": self.ticketInfoForPassengerForm["train_location"],
"seatDetailType": "", # 开始需要选择座位,但是目前12306不支持自动选择作为,那这个参数为默认
"roomType": "00", # 好像是根据一个id来判断选中的,两种 第一种是00,第二种是10,但是我在12306的页面没找到该id,目前写死是00,不知道会出什么错
"dwAll": "N",
"whatsSelect": 1,
"_json_at": "",
"randCode": "",
"choose_seats": "",
"REPEAT_SUBMIT_TOKEN": self.token,
}
return data
def sendConfirmSingleForQueue(self):
"""
# 模拟查询当前的列车排队人数的方法
# 返回信息组成的提示字符串
:return:
"""
data = self.data_par()
checkQueueOrderUrl = self.session.urls["checkQueueOrderUrl"]
try:
if self.is_node_code:
print(u"正在使用自动识别验证码功能")
for i in range(3):
randCode = getRandCode(is_auto_code=True, auto_code_type=2)
checkcode = checkRandCodeAnsyn(self.session, randCode, self.token)
if checkcode == 'TRUE':
print(u"验证码通过,正在提交订单")
data['randCode'] = randCode
break
else:
print (u"验证码有误, {0}次尝试重试".format(i + 1))
print(u"验证码超过限定次数3次,放弃此次订票机会!")
else:
print(u"不需要验证码")
time.sleep(self.ifShowPassCodeTime)
checkQueueOrderResult = self.session.httpClint.send(checkQueueOrderUrl, data)
if "status" in checkQueueOrderResult and checkQueueOrderResult["status"]:
c_data = checkQueueOrderResult["data"] if "data" in checkQueueOrderResult else {}
if 'submitStatus' in c_data and c_data['submitStatus'] is True:
qow = queryOrderWaitTime(self.session)
qow.sendQueryOrderWaitTime()
else:
if 'errMsg' in c_data and c_data['errMsg']:
print(u"提交订单失败,{0}".format(c_data['errMsg']))
else:
print(c_data)
print(u'订票失败!很抱歉,请重试提交预订功能!')
elif "messages" in checkQueueOrderResult and checkQueueOrderResult["messages"]:
print(u"提交订单失败,错误信息: " + checkQueueOrderResult["messages"])
else:
print(u"提交订单中,请耐心等待:" + checkQueueOrderResult["message"])
except ValueError:
print(u"接口 {} 无响应".format(checkQueueOrderUrl))
================================================
FILE: inter/ConfirmSingleForQueueAsys.py
================================================
# coding=utf-8
import json
import urllib
from collections import OrderedDict
from inter.QueryOrderWaitTime import queryOrderWaitTime
class confirmSingleForQueueAsys:
"""
订单快读排队
"""
def __init__(self,
session,
passengerTicketStr,
oldPassengerStr,
result,
randCode="",
):
self.session = session
self.passengerTicketStr = passengerTicketStr
self.oldPassengerStr = oldPassengerStr
self.result = result if isinstance(result, str) else str(result)
self.randCode = randCode
def data_par(self):
"""
字段说明
passengerTicketStr 乘客乘车代码
oldPassengerStr 乘客编号代码
randCode 填空
purpose_codes 学生还是成人
key_check_isChange autoSubmitOrderRequest返回的result字段做切割即可
leftTicketStr autoSubmitOrderRequest返回的result字段做切割即可
train_location autoSubmitOrderRequest返回的result字段做切割即可
choose_seats
seatDetailType
_json_att
:return:
"""
results = self.result.split("#")
key_check_isChange = results[1]
leftTicketStr = results[2]
train_location = results[0]
data = OrderedDict()
data["passengerTicketStr"] = self.passengerTicketStr
data["oldPassengerStr"] = self.oldPassengerStr
data["randCode"] = self.randCode
data["purpose_codes"] = "ADULT"
data["key_check_isChange"] = key_check_isChange
data["leftTicketStr"] = leftTicketStr
data["train_location"] = train_location
data["choose_seats"] = ""
data["seatDetailType"] = ""
data["_json_att"] = ""
return data
def sendConfirmSingleForQueueAsys(self):
"""
请求订单快读排队接口
:return:
"""
urls = self.session.urls["confirmSingleForQueueAsys"]
data = self.data_par()
confirmSingleForQueueAsysResult = self.session.httpClint.send(urls, data)
if confirmSingleForQueueAsysResult.get("status", False) and confirmSingleForQueueAsysResult.get("data", False):
queueData = confirmSingleForQueueAsysResult.get("data", {})
if queueData.get("submitStatus", False):
qwt = queryOrderWaitTime(session=self.session)
qwt.sendQueryOrderWaitTime()
else:
print(queueData.get("errMsg", ""))
================================================
FILE: inter/GetPassCodeNewOrderAndLogin.py
================================================
# coding=utf-8
import base64
import copy
import random
def getPassCodeNewOrderAndLogin(session, imgType):
"""
下载验证码
:param session:
:param imgType: 下载验证码类型,login=登录验证码,其余为订单验证码
:return:
"""
if imgType == "login":
codeImgUrl = copy.deepcopy(session.urls["getCodeImg"])
codeImgUrl["req_url"] = codeImgUrl["req_url"].format(random.random())
else:
codeImgUrl = copy.deepcopy(session.urls["codeImgByOrder"])
codeImgUrl["req_url"] = codeImgUrl["req_url"].format(random.random())
print(u"下载验证码...")
img_path = './tkcode.png'
result = session.httpClint.send(codeImgUrl)
try:
if isinstance(result, dict):
print(u"下载验证码失败, 请手动检查是否ip被封,或者重试,请求地址:https://kyfw.12306.cn{}".format(codeImgUrl.get("req_url")))
return False
else:
print(u"下载验证码成功")
try:
with open(img_path, 'wb', encoding="utf-8") as img:
img.write(result)
except Exception:
with open(img_path, 'wb') as img:
img.write(result)
return result
except OSError:
print(u"验证码下载失败,可能ip被封,确认请手动请求: {0}".format(codeImgUrl))
def getPassCodeNewOrderAndLogin1(session, imgType):
"""
获取验证码2
:param session:
:param imgType:
:return:
"""
if imgType == "login":
codeImgUrl = copy.deepcopy(session.urls["getCodeImg1"])
codeImgUrl["req_url"] = codeImgUrl["req_url"].format(random.random())
else:
codeImgUrl = copy.deepcopy(session.urls["codeImgByOrder"])
codeImgUrl["req_url"] = codeImgUrl["req_url"].format(random.random())
print(u"下载验证码...")
img_path = './tkcode.png'
codeImgUrlRsp = session.httpClint.send(codeImgUrl)
if not isinstance(codeImgUrlRsp, str):
print("验证码获取失败")
return
result = eval(codeImgUrlRsp.split("(")[1].split(")")[0]).get("image")
try:
if isinstance(result, dict):
print(u"下载验证码失败, 请手动检查是否ip被封,或者重试,请求地址:https://kyfw.12306.cn{}".format(codeImgUrl.get("req_url")))
return False
else:
print(u"下载验证码成功")
try:
with open(img_path, 'wb', encoding="utf-8") as img:
img.write(result)
except Exception:
with open(img_path, 'wb') as img:
img.write(base64.b64decode(result))
return result
except OSError:
print(u"验证码下载失败,可能ip被封或者文件写入没权限")
if __name__ == '__main__':
pass
================================================
FILE: inter/GetPassengerDTOs.py
================================================
# coding=utf-8
import json
from config.TicketEnmu import ticket
from myException.PassengerUserException import PassengerUserException
import wrapcache
import TickerConfig
class getPassengerDTOs:
"""
获取乘客信息
:return:
"""
def __init__(self, selectObj, ticket_peoples=None, set_type=None, is_more_ticket_num=None):
"""
:param session: 登录实例
:param ticket_peoples: 乘客
:param set_type: 坐席
"""
if ticket_peoples is None:
ticket_peoples = []
self.session = selectObj
self.ticket_peoples = ticket_peoples
self.is_more_ticket_num = is_more_ticket_num
self.set_type = set_type
def sendGetPassengerDTOs(self):
getPassengerDTOsResult = self.session.httpClint.send(self.session.urls["get_passengerDTOs"], json.dumps({"_json_att": ""}))
if getPassengerDTOsResult.get("data", False) and getPassengerDTOsResult["data"].get("normal_passengers", False):
normal_passengers = getPassengerDTOsResult['data']['normal_passengers']
_normal_passenger = [normal_passengers[i] for i in range(len(normal_passengers)) if
normal_passengers[i]["passenger_name"] in self.ticket_peoples]
return _normal_passenger if _normal_passenger else [normal_passengers[0]] # 如果配置乘车人没有在账号,则默认返回第一个用户
else:
if getPassengerDTOsResult.get("data", False) and getPassengerDTOsResult['data'].get("exMsg", False):
print(getPassengerDTOsResult['data'].get("exMsg", False))
elif getPassengerDTOsResult.get('messages', False):
print(getPassengerDTOsResult.get('messages', False))
else:
print(u"警告:您的账号可能买票有问题,获取不到联系人,请测试是否能正常下单,在捡漏或者购票!!!")
print(u"警告:您的账号可能买票有问题,获取不到联系人,请测试是否能正常下单,在捡漏或者购票!!!")
print(u"警告:您的账号可能买票有问题,获取不到联系人,请测试是否能正常下单,在捡漏或者购票!!!")
# raise PassengerUserException(ticket.DTO_NOT_FOUND)
def getPassengerTicketStr(self, set_type):
"""
获取getPassengerTicketStr 提交对应的代号码
:param str: 坐席
:return:
"""
passengerTicketStr = {
'一等座': 'M',
'特等座': 'P',
'二等座': 'O',
'商务座': 9,
'硬座': 1,
'无座': 1,
'软座': 2,
'软卧': 4,
'硬卧': 3,
}
return str(passengerTicketStr[set_type.replace(' ', '')])
def getPassengerTicketStrListAndOldPassengerStr(self, secretStr, secretList):
"""
获取提交车次人内容格式
passengerTicketStr O,0,1,文贤平,1,43052419950223XXXX,15618715583,N_O,0,1,梁敏,1,43052719920118XXXX,,N
oldPassengerStr 文贤平,1,43052719920118XXXX,1_梁敏,1,43052719920118XXXX,1
ps: 如果is_more_ticket打开了的话,那就是读取联系人列表里面前符合车次数量的前几个联系人
:return:
"""
passengerTicketStrList = []
oldPassengerStr = []
tickers = []
set_type = ""
if wrapcache.get("user_info"): # 如果缓存中有联系人方式,则读取缓存中的联系人
user_info = wrapcache.get("user_info")
print(u"使用缓存中查找的联系人信息")
else:
user_info = self.sendGetPassengerDTOs()
wrapcache.set("user_info", user_info, timeout=9999999)
if not user_info:
raise PassengerUserException(ticket.DTO_NOT_IN_LIST)
if len(user_info) < self.is_more_ticket_num: # 如果乘车人填错了导致没有这个乘车人的话,可能乘车人数会小于自动乘车人
self.is_more_ticket_num = len(user_info)
if secretStr:
set_type = self.getPassengerTicketStr(self.set_type)
if self.is_more_ticket_num is 1:
passengerTicketStrList.append(
'0,' + user_info[0]['passenger_type'] + "," + user_info[0][
"passenger_name"] + "," +
user_info[0]['passenger_id_type_code'] + "," + user_info[0]['passenger_id_no'] + "," +
user_info[0]['mobile_no'] + ',N,' + user_info[0]["allEncStr"])
oldPassengerStr.append(
user_info[0]['passenger_name'] + "," + user_info[0]['passenger_id_type_code'] + "," +
user_info[0]['passenger_id_no'] + "," + user_info[0]['passenger_type'] + '_')
else:
for i in range(self.is_more_ticket_num):
passengerTicketStrList.append(
'0,' + user_info[i]['passenger_type'] + "," + user_info[i][
"passenger_name"] + "," + user_info[i]['passenger_id_type_code'] + "," + user_info[i][
'passenger_id_no'] + "," + user_info[i]['mobile_no'] + ',N,' + user_info[i]["allEncStr"] + '_' + set_type)
oldPassengerStr.append(
user_info[i]['passenger_name'] + "," + user_info[i]['passenger_id_type_code'] + "," +
user_info[i]['passenger_id_no'] + "," + user_info[i]['passenger_type'] + '_')
elif secretList:
"""
候补订单有多少个联系人,就候补多少个联系人了,没有优先提交之说
1#XXXX#1#***************77X#bf6ae40d3655ae7eff005ee21d95876b38ab97a8031b464bc2f74a067e3ec957;
"""
for user in user_info:
tickers.append(f"1#{user['passenger_name']}#1#{user['passenger_id_no']}#{user['allEncStr']};")
return {
"passengerTicketStrList": set_type + "," + ",".join(passengerTicketStrList),
"passengerTicketStrByAfterLate": "".join(tickers),
"oldPassengerStr": "".join(oldPassengerStr),
"code": ticket.SUCCESS_CODE,
"set_type": set_type,
"status": True,
"user_info": user_info,
}
================================================
FILE: inter/GetQueueCount.py
================================================
# coding=utf-8
import datetime
import sys
import time
from collections import OrderedDict
import wrapcache
import TickerConfig
from config.TicketEnmu import ticket
from config.emailConf import sendEmail
from config.serverchanConf import sendServerChan
from config.urlConf import urls
from inter.ConfirmSingleForQueue import confirmSingleForQueue
from myException.ticketIsExitsException import ticketIsExitsException
def conversion_int(str):
return int(str)
class getQueueCount:
def __init__(self, session, is_need_code, ifShowPassCodeTime, set_type, station_dates, train_no, ticket_peoples,
ticketInfoForPassengerForm, token, oldPassengerStr, passengerTicketStrList):
self.station_dates = station_dates
self.session = session
self.is_need_code = is_need_code
self.ifShowPassCodeTime = ifShowPassCodeTime
self.set_type = set_type
self.train_no = train_no
self.ticket_peoples = ticket_peoples
self.ticket_black_list = {}
self.ticketInfoForPassengerForm = ticketInfoForPassengerForm
self.token = token
self.oldPassengerStr = oldPassengerStr
self.passengerTicketStrList = passengerTicketStrList
def data_par(self):
"""
参数结构
自动提交代码接口-autoSubmitOrderRequest
- 字段说明
- secretStr 车票代码
- train_date 乘车日期
- tour_flag 乘车类型
- purpose_codes 学生还是成人
- query_from_station_name 起始车站
- query_to_station_name 结束车站
- cancel_flag 默认2,我也不知道干嘛的
- bed_level_order_num 000000000000000000000000000000
- passengerTicketStr 乘客乘车代码
- oldPassengerStr 乘客编号代码
:return:
"""
if sys.version_info.major is 2:
new_train_date = filter(None, str(time.asctime(time.strptime(self.station_dates, "%Y-%m-%d"))).split(" "))
else:
new_train_date = list(filter(None, str(time.asctime(time.strptime(self.station_dates, "%Y-%m-%d"))).split(" ")))
data = OrderedDict()
data['train_date'] = "{0} {1} {2} {3} 00:00:00 GMT+0800 (中国标准时间)".format(
new_train_date[0],
new_train_date[1],
new_train_date[2] if len(new_train_date[2]) is 2 else f"0{new_train_date[2]}",
new_train_date[4],
),
data['train_no'] = self.ticketInfoForPassengerForm['queryLeftTicketRequestDTO']['train_no'],
data['stationTrainCode'] = self.ticketInfoForPassengerForm['queryLeftTicketRequestDTO'][
'station_train_code'],
data['seatType'] = self.set_type,
data['fromStationTelecode'] = self.ticketInfoForPassengerForm['queryLeftTicketRequestDTO'][
'from_station'],
data['toStationTelecode'] = self.ticketInfoForPassengerForm['queryLeftTicketRequestDTO']['to_station'],
data['leftTicket'] = self.ticketInfoForPassengerForm['leftTicketStr'],
data['purpose_codes'] = self.ticketInfoForPassengerForm['purpose_codes'],
data['train_location'] = self.ticketInfoForPassengerForm['train_location'],
data['REPEAT_SUBMIT_TOKEN'] = self.token,
return data
def sendGetQueueCount(self):
"""
# 模拟查询当前的列车排队人数的方法
# 返回信息组成的提示字符串
:return:
"""
getQueueCountResult = self.session.httpClint.send(self.session.urls["getQueueCountUrl"], self.data_par())
if "status" in getQueueCountResult and getQueueCountResult["status"] is True:
if "countT" in getQueueCountResult["data"]:
ticket = getQueueCountResult["data"]["ticket"]
ticket_split = sum(map(conversion_int, ticket.split(","))) if ticket.find(",") != -1 else ticket
countT = getQueueCountResult["data"]["countT"]
if int(ticket_split) is 0:
wrapcache.set(key=self.train_no, value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
print(f"排队失败,当前余票数还剩: {ticket_split} 张")
return
print(u"排队成功, 你排在: {1}位, 当前余票还剩余: {0} 张".format(ticket_split, countT))
csf = confirmSingleForQueue(self.session, self.ifShowPassCodeTime, self.is_need_code, self.token,
self.set_type, self.ticket_peoples, self.ticketInfoForPassengerForm,
self.oldPassengerStr, self.passengerTicketStrList)
csf.sendConfirmSingleForQueue()
# else:
# print(u"当前排队人数: {1} 当前余票还剩余:{0} 张,继续排队中".format(ticket_split, countT))
else:
print(u"排队发现未知错误{0},将此列车 {1}加入小黑屋".format(getQueueCountResult, self.train_no))
wrapcache.set(key=self.train_no, value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
elif "messages" in getQueueCountResult and getQueueCountResult["messages"]:
print(u"排队异常,错误信息:{0}, 将此列车 {1}加入小黑屋".format(getQueueCountResult["messages"][0], self.train_no))
wrapcache.set(key=self.train_no, value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
else:
if "validateMessages" in getQueueCountResult and getQueueCountResult["validateMessages"]:
print(str(getQueueCountResult["validateMessages"]))
wrapcache.set(key=self.train_no, value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
else:
print(u"未知错误 {0}".format("".join(getQueueCountResult)))
class queryQueueByAfterNate:
def __init__(self, session):
"""
候补排队
:param session:
"""
self.session = session
def sendQueryQueueByAfterNate(self):
for i in range(10):
queryQueueByAfterNateRsp = self.session.httpClint.send(urls.get("queryQueue"))
if not queryQueueByAfterNateRsp.get("status"):
print("".join(queryQueueByAfterNateRsp.get("messages")) or queryQueueByAfterNateRsp.get("validateMessages"))
time.sleep(1)
else:
sendEmail(ticket.WAIT_ORDER_SUCCESS)
sendServerChan(ticket.WAIT_ORDER_SUCCESS)
raise ticketIsExitsException(ticket.WAIT_AFTER_NATE_SUCCESS)
if __name__ == '__main__':
new_train_date = list(filter(None, str(time.asctime(time.strptime("2019-10-07", "%Y-%m-%d"))).split(" ")))
print(new_train_date)
train_date = "{0} {1} {2} {3} 00:00:00 GMT+0800 (中国标准时间)".format(
new_train_date[0],
new_train_date[1],
new_train_date[2] if len(new_train_date[2]) is 2 else f"0{new_train_date[2]}",
new_train_date[4],
)
print(train_date)
================================================
FILE: inter/GetQueueCountAsync.py
================================================
import TickerConfig
[]# coding=utf-8
import datetime
import sys
import time
from collections import OrderedDict
import wrapcache
from inter.ConfirmSingleForQueueAsys import confirmSingleForQueueAsys
class getQueueCountAsync:
"""
排队
"""
def __init__(self,
session,
train_no,
stationTrainCode,
fromStationTelecode,
toStationTelecode,
leftTicket,
set_type,
users,
station_dates,
passengerTicketStr,
oldPassengerStr,
result,
ifShowPassCodeTime):
self.train_no = train_no
self.session = session
self.stationTrainCode = stationTrainCode
self.fromStationTelecode = fromStationTelecode
self.toStationTelecode = toStationTelecode
self.set_type = set_type
self.leftTicket = leftTicket
self.users = users
self.station_dates = station_dates
self.passengerTicketStr = passengerTicketStr
self.oldPassengerStr = oldPassengerStr
self.result = result
self.ifShowPassCodeTime=ifShowPassCodeTime
def data_par(self):
"""
- 字段说明
- train_date 时间
- train_no 列车编号,查询代码里面返回
- stationTrainCode 列车编号
- seatType 对应坐席
- fromStationTelecode 起始城市
- toStationTelecode 到达城市
- leftTicket 查询代码里面返回
- purpose_codes 学生还是成人
- _json_att 没啥卵用,还是带上吧
:return:
"""
if sys.version_info.major is 2:
new_train_date = filter(None, str(time.asctime(time.strptime(self.station_dates, "%Y-%m-%d"))).split(" "))
else:
new_train_date = list(filter(None, str(time.asctime(time.strptime(self.station_dates, "%Y-%m-%d"))).split(" ")))
data = OrderedDict()
data['train_date'] = "{0} {1} {2} {3} 00:00:00 GMT+0800 (中国标准时间)".format(
new_train_date[0],
new_train_date[1],
new_train_date[2] if len(new_train_date[2]) is 2 else f"0{new_train_date[2]}",
new_train_date[4],
time.strftime("%H:%M:%S", time.localtime(time.time()))
),
data["train_no"] = self.train_no
data["stationTrainCode"] = self.stationTrainCode
data["seatType"] = self.set_type
data["fromStationTelecode"] = self.fromStationTelecode
data["toStationTelecode"] = self.toStationTelecode
data["leftTicket"] = self.leftTicket
data["purpose_codes"] = "ADULT"
data["_json_att"] = ""
return data
def conversion_int(self, str):
return int(str)
def sendGetQueueCountAsync(self):
"""
请求排队接口
:return:
"""
urls = self.session.urls["getQueueCountAsync"]
data = self.data_par()
getQueueCountAsyncResult = self.session.httpClint.send(urls, data)
if getQueueCountAsyncResult.get("status", False) and getQueueCountAsyncResult.get("data", False):
if "status" in getQueueCountAsyncResult and getQueueCountAsyncResult["status"] is True:
if "countT" in getQueueCountAsyncResult["data"]:
ticket_data = getQueueCountAsyncResult["data"]["ticket"]
ticket_split = sum(map(self.conversion_int, ticket_data.split(","))) if ticket_data.find(
",") != -1 else ticket_data
if int(ticket_split) is 0:
# 增加余票数为0时,将车次加入小黑屋
wrapcache.set(key=self.train_no, value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
print(f"排队失败,当前余票数为{ticket_split}张")
return
print(u"排队成功, 当前余票还剩余: {0} 张".format(ticket_split))
c = confirmSingleForQueueAsys(session=self.session,
passengerTicketStr=self.passengerTicketStr,
oldPassengerStr=self.oldPassengerStr,
result=self.result,)
print(u"验证码提交安全期,等待{}MS".format(self.ifShowPassCodeTime))
time.sleep(self.ifShowPassCodeTime)
c.sendConfirmSingleForQueueAsys()
else:
print(u"排队发现未知错误{0},将此列车 {1}加入小黑屋".format(getQueueCountAsyncResult, self.train_no))
wrapcache.set(key=self.train_no, value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
elif "messages" in getQueueCountAsyncResult and getQueueCountAsyncResult["messages"]:
print(u"排队异常,错误信息:{0}, 将此列车 {1}加入小黑屋".format(getQueueCountAsyncResult["messages"][0], self.train_no))
wrapcache.set(key=self.train_no, value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
else:
if "validateMessages" in getQueueCountAsyncResult and getQueueCountAsyncResult["validateMessages"]:
print(str(getQueueCountAsyncResult["validateMessages"]))
================================================
FILE: inter/GetRandCode.py
================================================
# coding=utf-8
from PIL import Image
from config.urlConf import urls
from myUrllib.httpUtils import HTTPClient
from verify.localVerifyCode import Verify
import TickerConfig
import os
if TickerConfig.AUTO_CODE_TYPE == 2:
v = Verify()
def getRandCode(is_auto_code, auto_code_type, result):
"""
识别验证码
:return: 坐标
"""
try:
if is_auto_code:
if auto_code_type == 1:
print(u"打码兔已关闭, 如需使用自动识别,请使用如果平台 auto_code_type == 2")
return
elif auto_code_type == 2:
Result = v.verify(result)
return codexy(Ofset=Result, is_raw_input=False)
elif auto_code_type == 3:
print("您已设置使用云打码,但是服务器资源有限,请尽快改为本地打码" if "CAPTCHALOCAL" not in os.environ else "已设置本地打码服务器")
http = HTTPClient(0)
Result = http.send(urls.get("autoVerifyImage"), {"imageFile": result})
if Result and Result.get("code") is 0:
return codexy(Ofset=Result.get("data"), is_raw_input=False)
else:
img = Image.open('./tkcode.png')
img.show()
return codexy()
except Exception as e:
print(e)
def codexy(Ofset=None, is_raw_input=True):
"""
获取验证码
:return: str
"""
if is_raw_input:
print(u"""
*****************
| 1 | 2 | 3 | 4 |
*****************
| 5 | 6 | 7 | 8 |
*****************
""")
print(u"验证码分为8个,对应上面数字,例如第一和第二张,输入1, 2 如果开启cdn查询的话,会冲掉提示,直接鼠标点击命令行获取焦点,输入即可,不要输入空格")
print(u"如果是linux无图形界面,请使用自动打码,is_auto_code: True")
print(u"如果没有弹出验证码,请手动双击根目录下的tkcode.png文件")
Ofset = input(u"输入对应的验证码: ")
if isinstance(Ofset, list):
select = Ofset
else:
Ofset = Ofset.replace(",", ",")
select = Ofset.split(',')
post = []
offsetsX = 0 # 选择的答案的left值,通过浏览器点击8个小图的中点得到的,这样基本没问题
offsetsY = 0 # 选择的答案的top值
for ofset in select:
if ofset == '1':
offsetsY = 77
offsetsX = 40
elif ofset == '2':
offsetsY = 77
offsetsX = 112
elif ofset == '3':
offsetsY = 77
offsetsX = 184
elif ofset == '4':
offsetsY = 77
offsetsX = 256
elif ofset == '5':
offsetsY = 149
offsetsX = 40
elif ofset == '6':
offsetsY = 149
offsetsX = 112
elif ofset == '7':
offsetsY = 149
offsetsX = 184
elif ofset == '8':
offsetsY = 149
offsetsX = 256
else:
pass
post.append(offsetsX)
post.append(offsetsY)
randCode = str(post).replace(']', '').replace('[', '').replace("'", '').replace(' ', '')
print(u"验证码识别坐标为{0}".format(randCode))
return randCode
================================================
FILE: inter/GetRepeatSubmitToken.py
================================================
# coding=utf-8
import json
import re
class getRepeatSubmitToken:
def __init__(self, session):
self.session = session
def sendGetRepeatSubmitToken(self):
"""
获取提交车票请求token
:return: token
"""
initdc_url = self.session.urls["initdc_url"]
initdc_result = self.session.httpClint.send(initdc_url, )
token_name = re.compile(r"var globalRepeatSubmitToken = '(\S+)'")
ticketInfoForPassengerForm_name = re.compile(r'var ticketInfoForPassengerForm=(\{.+\})?')
order_request_params_name = re.compile(r'var orderRequestDTO=(\{.+\})?')
token = re.search(token_name, initdc_result).group(1)
re_tfpf = re.findall(ticketInfoForPassengerForm_name, initdc_result)
re_orp = re.findall(order_request_params_name, initdc_result)
if re_tfpf:
ticketInfoForPassengerForm = json.loads(re_tfpf[0].replace("'", '"'))
else:
ticketInfoForPassengerForm = ""
if re_orp:
order_request_params = json.loads(re_orp[0].replace("'", '"'))
else:
order_request_params = ""
return {
"token": token,
"ticketInfoForPassengerForm": ticketInfoForPassengerForm,
"order_request_params": order_request_params,
"session": self.session
}
================================================
FILE: inter/GetSuccessRate.py
================================================
from collections import OrderedDict
from config.urlConf import urls
import TickerConfig
from inter.SubmitOrderRequest import submitOrderRequestByAfterNate
class getSuccessRate:
def __init__(self, session, secretList):
"""
获取成功信息
"""
self.secretList = secretList
self.session = session
def data_apr(self):
"""
secretList 9vqa9%2B%2F%2Fsdozmm22hpSeDTGqRUwSuA2D0r%2BmU%2BLZj7MK7CDuf5Ep1xpxl4Dyxfmoah%2BaB9TZSesU%0AkxBbo5oNgR1vqMfvq66VP0T7tpQtH%2BbVGBz1FolZG8jDD%2FHqnz%2FnvdBP416Og6WGS14O%2F3iBSwT8%0AkRPsNF0Vq0U082g0tlJtP%2BPn7TzW3z7TDCceMJIjFcfEOA%2BW%2BuK%2Bpy6jCQMv0TmlkXf5aKcGnE02%0APuv4I8nF%2BOWjWzv9CrJyiCZiWaXd%2Bi7p69V3a9dhF787UgS660%2BqKRFB4RLwAfic3MkAlfpGWhMY%0ACfARVQ%3D%3D#O
_json_att
候补一次只能补一个座位,默认取TICKET_TYPE第一个
:return:
"""
ticker = TickerConfig.PASSENGER_TICKER_STR.get(TickerConfig.SET_TYPE[0])
data = OrderedDict()
data["successSecret"] = f"{self.secretList}#{ticker}"
data["_json_att"] = ""
return data
def sendSuccessRate(self):
successRateRsp = self.session.httpClint.send(urls.get("getSuccessRate"), self.data_apr())
if not successRateRsp.get("status"):
print("".join(successRateRsp.get("messages")) or successRateRsp.get("validateMessages"))
return
flag = successRateRsp.get("data", {}).get("flag")[0]
train_no = flag.get("train_no")
print(f"准备提交候补订单,{flag.get('info')}")
submit = submitOrderRequestByAfterNate(self.session, self.secretList, train_no)
submit.sendSubmitOrderRequest()
================================================
FILE: inter/LiftTicketInit.py
================================================
# coding=utf-8
import re
class liftTicketInit:
def __init__(self, session):
self.session = session
def reqLiftTicketInit(self):
"""
请求抢票页面
:return:
"""
urls = self.session.urls["left_ticket_init"]
# 获取初始化的结果
result = self.session.httpClint.send(urls)
# 用正则表达式查出CLeftTicketUrl的值
matchObj = re.search('var CLeftTicketUrl = \'(.*)\'', result, re.M|re.I);
if matchObj:
# 如果有值,替换queryUrl
self.session.queryUrl = matchObj.group(1)
return {
"status": True
}
================================================
FILE: inter/LoginAysnSuggest.py
================================================
# coding=utf-8
from config.urlConf import urls
def loginAysnSuggest(session, username, password):
"""
登录接口
ps: 不需要验证码
:return:
"""
loginAysnSuggestUrls = urls.get("loginAysnSuggest")
data = {
"loginUserDTO.user_name": username,
"userDTO.password": password
}
loginAysnSuggestRsp = session.httpClint.send(urls=loginAysnSuggestUrls, data=data)
if loginAysnSuggestRsp and loginAysnSuggestRsp.get("httpstatus") is 200 and loginAysnSuggestRsp.get("data", {}).get("loginCheck") == "Y":
print(u"登录成功")
else:
print(u"登录失败, {0} {1}".format("".join(loginAysnSuggestRsp.get("messages")), loginAysnSuggestRsp.get("validateMessages")))
================================================
FILE: inter/LoginConf.py
================================================
# coding=utf-8
from config.urlConf import urls
def loginConf(session):
"""
判断登录是否需要验证码
:param session:
:return:
"""
loginConfUrl = urls.get("loginConf")
loginConfRsp = session.httpClint.send(urls=loginConfUrl, data={})
if loginConfRsp and loginConfRsp.get("data", {}).get("is_login_passCode") == "N":
print(u"不需要验证码")
return False
else:
print(u"需要验证码")
return True
if __name__ == '__main__':
pass
================================================
FILE: inter/PassengerInitApi.py
================================================
import datetime
import wrapcache
import TickerConfig
from config.urlConf import urls
from inter.ConfirmHB import confirmHB
class passengerInitApi:
def __init__(self, session, secretList, tickerNo):
"""
获取候补信息
"""
self.secretList = secretList
self.tickerNo = tickerNo
self.session = session
def sendPassengerInitApi(self):
passengerInitApiRsp = self.session.httpClint.send(urls.get("passengerInitApi"))
if not passengerInitApiRsp.get("status"):
print("".join(passengerInitApiRsp.get("messages")) or passengerInitApiRsp.get("validateMessages"))
return
data = passengerInitApiRsp.get("data", {})
jzdhDateE = data.get("jzdhDateE")
if not data.get("jzdhHourE"):
wrapcache.set(key=f"hb{self.tickerNo}", value=datetime.datetime.now(),
timeout=TickerConfig.TICKET_BLACK_LIST_TIME * 60)
print(f"获取当前候补日期失败,原因: {data.get('jzdhHourE')}")
return
jzdhHourE = data.get("jzdhHourE").replace(":", "#")
jzdhDate = f"{jzdhDateE}#{jzdhHourE}"
print(f"当前候补日期为:{jzdhDateE} {jzdhHourE}")
confirm = confirmHB(self.secretList, self.session, self.tickerNo, jzdhDate)
confirm.sendChechFace()
================================================
FILE: inter/Query.py
================================================
# coding=utf-8
import copy
import random
import wrapcache
from config import urlConf
from config.TicketEnmu import ticket
from myUrllib.httpUtils import HTTPClient
from config.configCommon import seat_conf_2
import TickerConfig
class query:
"""
查询接口
"""
def __init__(self, selectObj, from_station, to_station, from_station_h, to_station_h, _station_seat, station_trains,
ticke_peoples_num, station_dates=None, ):
self.session = selectObj
self.httpClint = HTTPClient(TickerConfig.IS_PROXY)
self.httpClint.set_cookies(self.session.cookies)
self.urls = urlConf.urls
self.from_station = from_station
self.to_station = to_station
self.from_station_h = from_station_h
self.to_station_h = to_station_h
self.station_trains = station_trains
self._station_seat = _station_seat if isinstance(_station_seat, list) else list(_station_seat)
self.station_dates = station_dates if isinstance(station_dates, list) else list(station_dates)
self.ticket_black_list = dict()
self.ticke_peoples_num = ticke_peoples_num
def station_seat(self, index):
"""
获取车票对应坐席
:return:
"""
seat = {'商务座': 32,
'一等座': 31,
'二等座': 30,
'特等座': 25,
'软卧': 23,
'硬卧': 28,
'硬座': 29,
'无座': 26,
'动卧': 33,
}
return seat[index]
def check_is_need_train(self, ticket_info):
"""
判断车次是否为想要的车次,如果ticket_info为空,那么就不校验车次,直接返回True
:param ticket_info:
:return:
"""
if self.station_dates and self.station_trains:
return ticket_info[3] in self.station_trains
else:
return True
def sendQuery(self):
"""
查询
:return:
"""
if TickerConfig.IS_CDN == 1 and self.session.cdn_list:
self.httpClint.cdn = self.session.cdn_list[random.randint(4, len(self.session.cdn_list) - 1)]
for station_date in self.station_dates:
select_url = copy.copy(self.urls["select_url"])
select_url["req_url"] = select_url["req_url"].format(station_date, self.from_station, self.to_station,
self.session.queryUrl)
station_ticket = self.httpClint.send(select_url)
value = station_ticket.get("data", "")
if not value:
print(u'{0}-{1} 车次坐席查询为空,查询url: https://kyfw.12306.cn{2}, 可以手动查询是否有票'.format(
self.from_station_h,
self.to_station_h,
select_url["req_url"]))
else:
result = value.get('result', [])
if result:
for i in value['result']:
ticket_info = i.split('|')
if self.session.flag:
print(f"车次:{ticket_info[3]} 出发站:{self.from_station_h} 到达站:{self.to_station_h} 历时:{ticket_info[10]}"
f" 商务/特等座:{ticket_info[32] or '--'}"
f" 一等座:{ticket_info[31] or '--'}"
f" 二等座:{ticket_info[30] or '--'}"
f" 动卧:{ticket_info[33] or '--'}"
f" 硬卧:{ticket_info[28] or '--'}"
f" 软座:{ticket_info[23] or '--'}"
f" 硬座:{ticket_info[29] or '--'}"
f" 无座:{ticket_info[26] or '--'}"
f" {ticket_info[1] or '--'}")
if ticket_info[1] == "预订" and self.check_is_need_train(ticket_info): # 筛选未在开始时间内的车次
for j in self._station_seat:
is_ticket_pass = ticket_info[j]
if ticket_info[11] == "Y":
if is_ticket_pass != '' and is_ticket_pass != '无' and is_ticket_pass != '*': # 过滤有效目标车次
secretStr = ticket_info[0]
train_no = ticket_info[2]
query_from_station_name = ticket_info[6]
query_to_station_name = ticket_info[7]
train_location = ticket_info[15]
stationTrainCode = ticket_info[3]
leftTicket = ticket_info[12]
start_time = ticket_info[8]
arrival_time = ticket_info[9]
distance_time = ticket_info[10]
print(start_time, arrival_time, distance_time)
seat = j
try:
ticket_num = int(ticket_info[j])
except ValueError:
ticket_num = "有"
print(u'车次: {0} 始发车站: {1} 终点站: {2} {3}: {4}'.format(ticket_info[3],
self.from_station_h,
self.to_station_h,
seat_conf_2[j],
ticket_num))
if seat_conf_2[j] == "无座" and ticket_info[3][0] in ["G", "D", "C"]:
seat = 30 # GD开头的无座直接强制改为二等座车次
if wrapcache.get(train_no):
print(ticket.QUERY_IN_BLACK_LIST.format(train_no))
continue
else:
if ticket_num != "有" and self.ticke_peoples_num > ticket_num:
if TickerConfig.IS_MORE_TICKET:
print(
u"余票数小于乘车人数,当前余票数: {}, 删减人车人数到: {}".format(ticket_num, ticket_num))
is_more_ticket_num = ticket_num
else:
print(u"余票数小于乘车人数,当前设置不提交,放弃此次提交机会")
continue
else:
print(u"设置乘车人数为: {}".format(self.ticke_peoples_num))
is_more_ticket_num = self.ticke_peoples_num
print(ticket.QUERY_C)
return {
"secretStr": secretStr,
"train_no": train_no,
"stationTrainCode": stationTrainCode,
"train_date": station_date,
"query_from_station_name": query_from_station_name,
"query_to_station_name": query_to_station_name,
"seat": seat,
"leftTicket": leftTicket,
"train_location": train_location,
"code": ticket.SUCCESS_CODE,
"is_more_ticket_num": is_more_ticket_num,
"cdn": self.httpClint.cdn,
"status": True,
}
elif is_ticket_pass == '无' and ticket_info[37] == "1" and TickerConfig.TICKET_TYPE is 2:
"""
is_ticket_pass如果有别的显示,但是可以候补,可以提issues提出来,附上query log,我将添加上
判断车次是否可以候补
目前的候补机制是只要一有候补位置,立马提交候补
"""
# 如果最后一位为1,则是可以候补的,不知道这些正确嘛?
nate = list(ticket_info[38])
if wrapcache.get(f"hb{ticket_info[2]}"):
continue
for set_type in TickerConfig.SET_TYPE:
if TickerConfig.PASSENGER_TICKER_STR[set_type] not in nate:
if ticket_info[3][0] in ["G", "D", "C"] and set_type in ["一等座", "特等座", "二等座", "商务座", "无座"]:
return {
"secretList": ticket_info[0],
"seat": [set_type],
"train_no": ticket_info[2],
"status": True,
"cdn": self.httpClint.cdn,
}
elif ticket_info[3][0] in ["T", "Z", "K"] and set_type in ["硬卧", "硬座", "无座", "软座", "软卧"]:
return {
"secretList": ticket_info[0],
"seat": [set_type],
"train_no": ticket_info[2],
"status": True,
"cdn": self.httpClint.cdn,
}
else:
print(u"车次配置信息有误,或者返回数据异常,请检查 {}".format(station_ticket))
self.session.flag = False
return {"code": ticket.FAIL_CODE, "status": False, "cdn": self.httpClint.cdn, }
if __name__ == "__main__":
q = query()
================================================
FILE: inter/QueryOrderWaitTime.py
================================================
# coding=utf-8
import copy
import time
from config.TicketEnmu import ticket
from config.emailConf import sendEmail
from config.serverchanConf import sendServerChan
from myException.ticketIsExitsException import ticketIsExitsException
from myException.ticketNumOutException import ticketNumOutException
class queryOrderWaitTime:
"""
排队
"""
def __init__(self, session):
self.session = session
def sendQueryOrderWaitTime(self):
"""
排队获取订单等待信息,每隔3秒请求一次,最高请求次数为20次!
:return:
"""
num = 1
while True:
num += 1
if num > ticket.OUT_NUM:
print(ticket.WAIT_OUT_NUM)
order_id = self.queryMyOrderNoComplete() # 排队失败,自动取消排队订单
if order_id:
self.cancelNoCompleteMyOrder(order_id)
break
try:
queryOrderWaitTimeUrl = copy.deepcopy(self.session.urls["queryOrderWaitTimeUrl"])
queryOrderWaitTimeUrl["req_url"] = queryOrderWaitTimeUrl["req_url"].format(int(round(time.time() * 1000)))
queryOrderWaitTimeResult = self.session.httpClint.send(queryOrderWaitTimeUrl)
except ValueError:
queryOrderWaitTimeResult = {}
if queryOrderWaitTimeResult:
if queryOrderWaitTimeResult.get("status", False):
data = queryOrderWaitTimeResult.get
gitextract_ltfh64ff/
├── .dockerignore
├── .github/
│ └── ISSUE_TEMPLATE/
│ ├── bug_report.md
│ └── feature_request.md
├── .gitignore
├── 12306.image.model.h5
├── Dockerfile
├── Dockerfile37
├── LICENSE
├── README.md
├── TickerConfig.py
├── UnitTest/
│ ├── TestAll.py
│ └── __init__.py
├── Update.md
├── __init__.py
├── agency/
│ ├── __init__.py
│ ├── agency_tools.py
│ ├── cdn_utils.py
│ └── proxy_list
├── cdn_list
├── config/
│ ├── AutoSynchroTime.py
│ ├── TicketEnmu.py
│ ├── __init__.py
│ ├── configCommon.py
│ ├── emailConf.py
│ ├── getCookie.py
│ ├── logger.py
│ ├── pushbearConf.py
│ ├── serverchanConf.py
│ └── urlConf.py
├── docker-compose.yml
├── docker_install_centos.sh
├── filter_cdn_list
├── init/
│ ├── __init__.py
│ ├── login.py
│ └── select_ticket_info.py
├── inter/
│ ├── AutoSubmitOrderRequest.py
│ ├── ChechFace.py
│ ├── CheckOrderInfo.py
│ ├── CheckRandCodeAnsyn.py
│ ├── CheckUser.py
│ ├── ConfirmHB.py
│ ├── ConfirmSingleForQueue.py
│ ├── ConfirmSingleForQueueAsys.py
│ ├── GetPassCodeNewOrderAndLogin.py
│ ├── GetPassengerDTOs.py
│ ├── GetQueueCount.py
│ ├── GetQueueCountAsync.py
│ ├── GetRandCode.py
│ ├── GetRepeatSubmitToken.py
│ ├── GetSuccessRate.py
│ ├── LiftTicketInit.py
│ ├── LoginAysnSuggest.py
│ ├── LoginConf.py
│ ├── PassengerInitApi.py
│ ├── Query.py
│ ├── QueryOrderWaitTime.py
│ ├── SubmitOrderRequest.py
│ └── __init__.py
├── model.v2.0.h5
├── myException/
│ ├── PassengerUserException.py
│ ├── UserPasswordException.py
│ ├── __init__.py
│ ├── balanceException.py
│ ├── ticketConfigException.py
│ ├── ticketIsExitsException.py
│ └── ticketNumOutException.py
├── myUrllib/
│ ├── MySocketUtils.py
│ ├── __init__.py
│ └── httpUtils.py
├── requirements-docker37.txt
├── requirements.txt
├── run.py
├── station_name.txt
├── tmp/
│ ├── __init__.py
│ └── log/
│ └── __init__.py
└── verify/
├── __init__.py
├── localVerifyCode.py
├── mlearn_for_image.py
└── pretreatment.py
SYMBOL INDEX (209 symbols across 47 files)
FILE: UnitTest/TestAll.py
function _set_header_default (line 16) | def _set_header_default():
class testAll (line 27) | class testAll(unittest.TestCase):
method testProxy (line 28) | def testProxy(self):
method testEmail (line 39) | def testEmail(self):
method testServerChan (line 52) | def testServerChan(self):
method testUserAgent (line 59) | def testUserAgent(self):
method testVerfyImage (line 69) | def testVerfyImage(self):
method testRemoteVerfy (line 82) | def testRemoteVerfy(self):
method testCdn (line 102) | def testCdn(self):
FILE: agency/agency_tools.py
class proxy (line 11) | class proxy:
method __init__ (line 12) | def __init__(self):
method get_proxy (line 16) | def get_proxy(self):
method filter_proxy (line 40) | def filter_proxy(self):
method get_filter_proxy (line 66) | def get_filter_proxy(self):
method main (line 86) | def main(self):
method setProxy (line 90) | def setProxy(self):
FILE: agency/cdn_utils.py
class CDNProxy (line 15) | class CDNProxy(threading.Thread):
method __init__ (line 16) | def __init__(self, cdns):
method run (line 24) | def run(self):
function open_cdn_file (line 38) | def open_cdn_file(cdnFile):
function sortCdn (line 55) | def sortCdn():
function filterCdn (line 68) | def filterCdn():
FILE: config/AutoSynchroTime.py
function autoSynchroTime (line 9) | def autoSynchroTime():
FILE: config/TicketEnmu.py
class ticket (line 5) | class ticket(object):
FILE: config/configCommon.py
function getNowTimestamp (line 42) | def getNowTimestamp():
function decMakeDir (line 46) | def decMakeDir(func):
function getWorkDir (line 59) | def getWorkDir():
function getTmpDir (line 79) | def getTmpDir():
function getLogDir (line 84) | def getLogDir():
function getCacheDir (line 89) | def getCacheDir():
function getVCodeDir (line 94) | def getVCodeDir():
function getVCodeImageFile (line 98) | def getVCodeImageFile(imageName):
function getCacheFile (line 102) | def getCacheFile(cacheType):
function checkSleepTime (line 106) | def checkSleepTime(session):
function checkDate (line 117) | def checkDate(station_dates):
FILE: config/emailConf.py
function sendEmail (line 10) | def sendEmail(msg):
FILE: config/getCookie.py
function getDrvicesID (line 10) | def getDrvicesID(session):
function request_device_id (line 55) | def request_device_id(session):
function request_alg_id (line 79) | def request_alg_id(session):
function _get_hash_code_params (line 89) | def _get_hash_code_params():
function _encode_data_str_v2 (line 183) | def _encode_data_str_v2(d):
function _encode_string (line 191) | def _encode_string(str):
FILE: config/logger.py
function setSuffix (line 14) | def setSuffix(s):
function getTodayDateStr (line 18) | def getTodayDateStr():
function setDateStr (line 21) | def setDateStr(s):
function isAnotherDay (line 25) | def isAnotherDay(s):
function getLogFile (line 29) | def getLogFile():
function log (line 36) | def log(msg, func = "info"):
FILE: config/pushbearConf.py
function sendPushBear (line 9) | def sendPushBear(msg):
FILE: config/serverchanConf.py
function sendServerChan (line 9) | def sendServerChan(msg):
FILE: init/login.py
class GoLogin (line 14) | class GoLogin:
method __init__ (line 15) | def __init__(self, session, is_auto_code, auto_code_type):
method auth (line 21) | def auth(self):
method codeCheck (line 30) | def codeCheck(self):
method baseLogin (line 51) | def baseLogin(self, user, passwd):
method getUserName (line 85) | def getUserName(self, uamtk):
method go_login (line 107) | def go_login(self):
FILE: init/select_ticket_info.py
class select (line 32) | class select:
method __init__ (line 36) | def __init__(self):
method get_ticket_info (line 54) | def get_ticket_info():
method station_table (line 82) | def station_table(self, from_station, to_station):
method call_login (line 108) | def call_login(self, auth=False):
method main (line 119) | def main(self):
FILE: inter/AutoSubmitOrderRequest.py
class autoSubmitOrderRequest (line 12) | class autoSubmitOrderRequest:
method __init__ (line 16) | def __init__(self, selectObj,
method data_par (line 42) | def data_par(self):
method sendAutoSubmitOrderRequest (line 72) | def sendAutoSubmitOrderRequest(self):
FILE: inter/ChechFace.py
class chechFace (line 11) | class chechFace:
method __init__ (line 12) | def __init__(self, selectObj, secretList, train_no):
method data_apr (line 20) | def data_apr(self):
method sendChechFace (line 33) | def sendChechFace(self):
FILE: inter/CheckOrderInfo.py
class checkOrderInfo (line 7) | class checkOrderInfo:
method __init__ (line 9) | def __init__(self, session, train_no, set_type, passengerTicketStrList...
method data_par (line 22) | def data_par(self):
method sendCheckOrderInfo (line 38) | def sendCheckOrderInfo(self):
FILE: inter/CheckRandCodeAnsyn.py
class checkRandCodeAnsyn (line 2) | class checkRandCodeAnsyn:
method __init__ (line 3) | def __init__(self, session, randCode, token):
method data_par (line 8) | def data_par(self):
method sendCheckRandCodeAnsyn (line 20) | def sendCheckRandCodeAnsyn(self):
FILE: inter/CheckUser.py
class checkUser (line 10) | class checkUser:
method __init__ (line 11) | def __init__(self, session):
method sendCheckUser (line 14) | def sendCheckUser(self):
FILE: inter/ConfirmHB.py
class confirmHB (line 7) | class confirmHB:
method __init__ (line 8) | def __init__(self, secretList, session, tickerNo, jzdhDate):
method data_apr (line 18) | def data_apr(self):
method sendChechFace (line 34) | def sendChechFace(self):
FILE: inter/ConfirmSingleForQueue.py
class confirmSingleForQueue (line 11) | class confirmSingleForQueue:
method __init__ (line 12) | def __init__(self, session, ifShowPassCodeTime, is_node_code, token, s...
method data_par (line 24) | def data_par(self):
method sendConfirmSingleForQueue (line 53) | def sendConfirmSingleForQueue(self):
FILE: inter/ConfirmSingleForQueueAsys.py
class confirmSingleForQueueAsys (line 9) | class confirmSingleForQueueAsys:
method __init__ (line 13) | def __init__(self,
method data_par (line 26) | def data_par(self):
method sendConfirmSingleForQueueAsys (line 58) | def sendConfirmSingleForQueueAsys(self):
FILE: inter/GetPassCodeNewOrderAndLogin.py
function getPassCodeNewOrderAndLogin (line 7) | def getPassCodeNewOrderAndLogin(session, imgType):
function getPassCodeNewOrderAndLogin1 (line 40) | def getPassCodeNewOrderAndLogin1(session, imgType):
FILE: inter/GetPassengerDTOs.py
class getPassengerDTOs (line 10) | class getPassengerDTOs:
method __init__ (line 15) | def __init__(self, selectObj, ticket_peoples=None, set_type=None, is_m...
method sendGetPassengerDTOs (line 28) | def sendGetPassengerDTOs(self):
method getPassengerTicketStr (line 46) | def getPassengerTicketStr(self, set_type):
method getPassengerTicketStrListAndOldPassengerStr (line 65) | def getPassengerTicketStrListAndOldPassengerStr(self, secretStr, secre...
FILE: inter/GetQueueCount.py
function conversion_int (line 17) | def conversion_int(str):
class getQueueCount (line 21) | class getQueueCount:
method __init__ (line 22) | def __init__(self, session, is_need_code, ifShowPassCodeTime, set_type...
method data_par (line 37) | def data_par(self):
method sendGetQueueCount (line 79) | def sendGetQueueCount(self):
class queryQueueByAfterNate (line 120) | class queryQueueByAfterNate:
method __init__ (line 121) | def __init__(self, session):
method sendQueryQueueByAfterNate (line 128) | def sendQueryQueueByAfterNate(self):
FILE: inter/GetQueueCountAsync.py
class getQueueCountAsync (line 14) | class getQueueCountAsync:
method __init__ (line 18) | def __init__(self,
method data_par (line 46) | def data_par(self):
method conversion_int (line 82) | def conversion_int(self, str):
method sendGetQueueCountAsync (line 85) | def sendGetQueueCountAsync(self):
FILE: inter/GetRandCode.py
function getRandCode (line 15) | def getRandCode(is_auto_code, auto_code_type, result):
function codexy (line 42) | def codexy(Ofset=None, is_raw_input=True):
FILE: inter/GetRepeatSubmitToken.py
class getRepeatSubmitToken (line 6) | class getRepeatSubmitToken:
method __init__ (line 7) | def __init__(self, session):
method sendGetRepeatSubmitToken (line 10) | def sendGetRepeatSubmitToken(self):
FILE: inter/GetSuccessRate.py
class getSuccessRate (line 9) | class getSuccessRate:
method __init__ (line 10) | def __init__(self, session, secretList):
method data_apr (line 17) | def data_apr(self):
method sendSuccessRate (line 31) | def sendSuccessRate(self):
FILE: inter/LiftTicketInit.py
class liftTicketInit (line 5) | class liftTicketInit:
method __init__ (line 6) | def __init__(self, session):
method reqLiftTicketInit (line 9) | def reqLiftTicketInit(self):
FILE: inter/LoginAysnSuggest.py
function loginAysnSuggest (line 5) | def loginAysnSuggest(session, username, password):
FILE: inter/LoginConf.py
function loginConf (line 5) | def loginConf(session):
FILE: inter/PassengerInitApi.py
class passengerInitApi (line 8) | class passengerInitApi:
method __init__ (line 9) | def __init__(self, session, secretList, tickerNo):
method sendPassengerInitApi (line 17) | def sendPassengerInitApi(self):
FILE: inter/Query.py
class query (line 12) | class query:
method __init__ (line 17) | def __init__(self, selectObj, from_station, to_station, from_station_h...
method station_seat (line 33) | def station_seat(self, index):
method check_is_need_train (line 50) | def check_is_need_train(self, ticket_info):
method sendQuery (line 61) | def sendQuery(self):
FILE: inter/QueryOrderWaitTime.py
class queryOrderWaitTime (line 12) | class queryOrderWaitTime:
method __init__ (line 17) | def __init__(self, session):
method sendQueryOrderWaitTime (line 20) | def sendQueryOrderWaitTime(self):
method queryMyOrderNoComplete (line 67) | def queryMyOrderNoComplete(self):
method initNoComplete (line 96) | def initNoComplete(self):
method cancelNoCompleteMyOrder (line 105) | def cancelNoCompleteMyOrder(self, sequence_no):
FILE: inter/SubmitOrderRequest.py
function time (line 13) | def time():
class submitOrderRequest (line 22) | class submitOrderRequest:
method __init__ (line 23) | def __init__(self, selectObj, secretStr, from_station, to_station, tra...
method data_apr (line 41) | def data_apr(self):
method sendSubmitOrderRequest (line 56) | def sendSubmitOrderRequest(self):
class submitOrderRequestByAfterNate (line 79) | class submitOrderRequestByAfterNate:
method __init__ (line 80) | def __init__(self, session, secretList, tickerNo):
method data_apr (line 90) | def data_apr(self):
method sendSubmitOrderRequest (line 104) | def sendSubmitOrderRequest(self, ):
FILE: myException/PassengerUserException.py
class PassengerUserException (line 1) | class PassengerUserException(Exception):
FILE: myException/UserPasswordException.py
class UserPasswordException (line 1) | class UserPasswordException(Exception):
FILE: myException/balanceException.py
class balanceException (line 1) | class balanceException(Exception):
FILE: myException/ticketConfigException.py
class ticketConfigException (line 1) | class ticketConfigException(Exception):
FILE: myException/ticketIsExitsException.py
class ticketIsExitsException (line 1) | class ticketIsExitsException(Exception):
FILE: myException/ticketNumOutException.py
class ticketNumOutException (line 1) | class ticketNumOutException(Exception):
FILE: myUrllib/MySocketUtils.py
function default_get_data (line 30) | def default_get_data():
function default_post_data (line 51) | def default_post_data():
class socketUtils (line 78) | class socketUtils:
method __init__ (line 79) | def __init__(self, host, port=80):
method connect_socket (line 84) | def connect_socket(self, host, port):
method close_s (line 96) | def close_s(self):
method recv_data (line 135) | def recv_data(self, r_data):
method get_cookie (line 146) | def get_cookie(recv_data):
method get_status_code (line 159) | def get_status_code(recv_data):
method get_rep_body (line 171) | def get_rep_body(recv_data):
FILE: myUrllib/httpUtils.py
function _set_header_default (line 14) | def _set_header_default():
function _set_user_agent (line 26) | def _set_user_agent():
class HTTPClient (line 37) | class HTTPClient(object):
method __init__ (line 39) | def __init__(self, is_proxy, cdnList=None):
method initS (line 54) | def initS(self):
method set_cookies (line 59) | def set_cookies(self, kwargs):
method get_cookies (line 69) | def get_cookies(self):
method del_cookies (line 76) | def del_cookies(self):
method del_cookies_by_key (line 83) | def del_cookies_by_key(self, key):
method setHeaders (line 90) | def setHeaders(self, headers):
method resetHeaders (line 94) | def resetHeaders(self):
method getHeadersHost (line 98) | def getHeadersHost(self):
method setHeadersHost (line 101) | def setHeadersHost(self, host):
method setHeadersUserAgent (line 105) | def setHeadersUserAgent(self):
method getHeadersUserAgent (line 108) | def getHeadersUserAgent(self):
method getHeadersReferer (line 111) | def getHeadersReferer(self):
method setHeadersReferer (line 114) | def setHeadersReferer(self, referer):
method cdn (line 119) | def cdn(self):
method cdn (line 123) | def cdn(self, cdn):
method send (line 126) | def send(self, urls, data=None, **kwargs):
FILE: run.py
function parser_arguments (line 6) | def parser_arguments(argv):
FILE: verify/localVerifyCode.py
function get_text (line 25) | def get_text(img, offset=0):
function base64_to_image (line 34) | def base64_to_image(base64_code):
class Verify (line 45) | class Verify:
method __init__ (line 46) | def __init__(self):
method loadTextModel (line 52) | def loadTextModel(self):
method loadImgModel (line 58) | def loadImgModel(self):
method verify (line 62) | def verify(self, fn):
FILE: verify/mlearn_for_image.py
function preprocess_input (line 16) | def preprocess_input(x):
function load_data (line 24) | def load_data():
function learn (line 41) | def learn():
function predict (line 74) | def predict(imgs):
function _predict (line 81) | def _predict(fn):
FILE: verify/pretreatment.py
function download_image (line 21) | def download_image():
function download_images (line 32) | def download_images():
function get_text (line 39) | def get_text(img, offset=0):
function avhash (line 44) | def avhash(im):
function phash (line 52) | def phash(im):
function _get_imgs (line 62) | def _get_imgs(img):
function get_imgs (line 70) | def get_imgs(img):
function pretreat (line 77) | def pretreat():
function load_data (line 89) | def load_data(path='data.npz'):
Condensed preview — 79 files, each showing path, character count, and a content snippet. Download the .json file or copy for the full structured content (332K chars).
[
{
"path": ".dockerignore",
"chars": 123,
"preview": "**/*.html\n**/*.pyc\n**/*.yaml\n**/*.log\n**/*~\n**/.DS_Store\n**/Thumbs.db\n*.png\n.idea/\n.git/\n.github/\n*.md\nUnitTest/\numl/\n*."
},
{
"path": ".github/ISSUE_TEMPLATE/bug_report.md",
"chars": 318,
"preview": "---\nname: Bug report\nabout: Create a report to help us improve\ntitle: ''\nlabels: ''\nassignees: ''\n\n---\n\n**描述问题**\n```\nA c"
},
{
"path": ".github/ISSUE_TEMPLATE/feature_request.md",
"chars": 595,
"preview": "---\nname: Feature request\nabout: Suggest an idea for this project\ntitle: ''\nlabels: ''\nassignees: ''\n\n---\n\n**Is your fea"
},
{
"path": ".gitignore",
"chars": 43,
"preview": "*.html\n*.pyc\n*.yaml\n*.log\n.idea/\ntkcode.png"
},
{
"path": "Dockerfile",
"chars": 778,
"preview": "FROM python:2.7.15\nWORKDIR /usr/src/app\nADD . /usr/src/app\n\nENV DEBIAN_FRONTEND noninteractive\nENV TZ Asia/Shanghai\n\n\n##"
},
{
"path": "Dockerfile37",
"chars": 1312,
"preview": "FROM python:3.7-slim-buster\n\nARG CDV=77.0.3865.40\n\nRUN sed -i 's/deb.debian.org/ftp.cn.debian.org/g' /etc/apt/sources.li"
},
{
"path": "LICENSE",
"chars": 1055,
"preview": "MIT License\n\nCopyright (c) 2017\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this so"
},
{
"path": "README.md",
"chars": 4460,
"preview": "### 12306 购票小助手\n#### python版本\n - [ ] 2.7.10 - 2.7.15\n - [x] 3.6 - 3.7.4\n - [ ] 2.7.9\n\n#### 已有功能\n - [x] 自动打码\n - [x] "
},
{
"path": "TickerConfig.py",
"chars": 3539,
"preview": "# -*- coding=utf-8 -*-\n# 关于软件使用配置说明,一定要看!!!\n# ps: 如果是候补车票,需要通过人证一致性核验的用户及激活的“铁路畅行”会员可以提交候补需求,请您按照操作说明在铁路12306app.上完成人证核验"
},
{
"path": "UnitTest/TestAll.py",
"chars": 18619,
"preview": "# coding=utf-8\nimport base64\nimport threading\nimport unittest\nfrom collections import OrderedDict\n\nimport requests\n\nimpo"
},
{
"path": "UnitTest/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "Update.md",
"chars": 3944,
"preview": "- 2017.5.13跟新\n - 增加登陆错误判断(密码错误&ip校验)\n - 修改queryOrderWaitTime,校验orderId字段bug,校验msg字段bug,校验messagesbug\n - 修改check"
},
{
"path": "__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "agency/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "agency/agency_tools.py",
"chars": 3214,
"preview": "# encoding=utf8\nimport os\nimport random\nimport socket\nimport time\n\nimport requests\nfrom bs4 import BeautifulSoup\n\n\nclass"
},
{
"path": "agency/cdn_utils.py",
"chars": 2643,
"preview": "# encoding=utf8\nimport datetime\nimport operator\nimport os\nimport requests\nfrom config import urlConf\nimport threading\nfr"
},
{
"path": "agency/proxy_list",
"chars": 20,
"preview": "119.101.114.196:9999"
},
{
"path": "cdn_list",
"chars": 26459,
"preview": "112.123.33.18\n112.28.196.75\n112.28.196.100\n112.29.227.103\n112.29.227.250\n112.29.227.106\n112.30.197.137\n112.28.196.54\n112"
},
{
"path": "config/AutoSynchroTime.py",
"chars": 1270,
"preview": "# coding=utf-8\nimport os\nimport platform\n\nimport ntplib\nimport datetime\n\n\ndef autoSynchroTime():\n \"\"\"\n 同步北京时间,执行时候"
},
{
"path": "config/TicketEnmu.py",
"chars": 1380,
"preview": "# coding=utf-8\nfrom enum import Enum\n\n\nclass ticket(object):\n QUERY_C = u\"查询到有余票,尝试提交订单\"\n QUERY_IN_BLACK_LIST = u\""
},
{
"path": "config/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "config/configCommon.py",
"chars": 3243,
"preview": "# -*- coding: utf-8 -*-\nimport datetime\nimport os\nimport random\nimport sys\nimport time\n\nfrom myException.ticketConfigExc"
},
{
"path": "config/emailConf.py",
"chars": 1379,
"preview": "# -*- coding: utf8 -*-\nimport socket\n__author__ = 'MR.wen'\nimport TickerConfig\nfrom email.header import Header\nfrom emai"
},
{
"path": "config/getCookie.py",
"chars": 6618,
"preview": "import json\nimport random\nimport re\nimport time\nimport os\nimport TickerConfig\nfrom config.urlConf import urls\n\n\ndef getD"
},
{
"path": "config/logger.py",
"chars": 1183,
"preview": "#coding: utf-8\n\nimport os\nimport time\nimport logging\n\nfrom config import configCommon\n\nlogger = None\nloggerHandler = Non"
},
{
"path": "config/pushbearConf.py",
"chars": 1033,
"preview": "# -*- coding: utf8 -*-\nimport TickerConfig\nfrom config.urlConf import urls\nfrom myUrllib.httpUtils import HTTPClient\n\nPU"
},
{
"path": "config/serverchanConf.py",
"chars": 1069,
"preview": "# -*- coding: utf8 -*-\nimport TickerConfig\nfrom config.urlConf import urls\nfrom myUrllib.httpUtils import HTTPClient\n\nPU"
},
{
"path": "config/urlConf.py",
"chars": 17580,
"preview": "# coding=utf-8\nimport random\nimport TickerConfig\nimport time\n\nurls = {\n \"auth\": { # 登录接口\n \"req_url\": \"/passpo"
},
{
"path": "docker-compose.yml",
"chars": 744,
"preview": "version: \"3\"\n\nservices:\n #抢票服务\n ticket:\n build:\n context: .\n dockerfile: ./Dockerfile"
},
{
"path": "docker_install_centos.sh",
"chars": 961,
"preview": "#!/bin/bash\n#author: MonsterTan\n#date: 2019-01-15\n#this is a script that can install automatically docker software by ce"
},
{
"path": "filter_cdn_list",
"chars": 8090,
"preview": "111.161.122.240\n112.90.135.96\n60.9.0.19\n61.162.100.102\n222.186.141.146\n221.235.187.129\n58.221.78.231\n113.16.212.251\n112."
},
{
"path": "init/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "init/login.py",
"chars": 4733,
"preview": "# -*- coding=utf-8 -*-\nimport copy\nimport time\nfrom collections import OrderedDict\nfrom time import sleep\nimport TickerC"
},
{
"path": "init/select_ticket_info.py",
"chars": 11962,
"preview": "# -*- coding=utf-8 -*-\nimport datetime\nimport random\nimport os\nimport socket\nimport sys\nimport threading\nimport time\nimp"
},
{
"path": "inter/AutoSubmitOrderRequest.py",
"chars": 5444,
"preview": "# coding=utf-8\nimport urllib\nfrom collections import OrderedDict\n\nfrom config.TicketEnmu import ticket\nfrom inter.CheckR"
},
{
"path": "inter/ChechFace.py",
"chars": 3052,
"preview": "import datetime\nimport urllib\nfrom collections import OrderedDict\nfrom config.urlConf import urls\nimport TickerConfig\nfr"
},
{
"path": "inter/CheckOrderInfo.py",
"chars": 3501,
"preview": "# coding=utf-8\nfrom collections import OrderedDict\nfrom inter.GetQueueCount import getQueueCount\nfrom inter.GetRepeatSub"
},
{
"path": "inter/CheckRandCodeAnsyn.py",
"chars": 742,
"preview": "# coding=utf-8\nclass checkRandCodeAnsyn:\n def __init__(self, session, randCode, token):\n self.session = sessio"
},
{
"path": "inter/CheckUser.py",
"chars": 1661,
"preview": "# coding=utf-8\nimport datetime\nimport random\nimport time\nimport wrapcache\nfrom config import configCommon\nfrom config.Ti"
},
{
"path": "inter/ConfirmHB.py",
"chars": 1579,
"preview": "from collections import OrderedDict\nfrom config.urlConf import urls\nimport TickerConfig\nfrom inter.GetQueueCount import "
},
{
"path": "inter/ConfirmSingleForQueue.py",
"chars": 4446,
"preview": "# coding=utf-8\nimport datetime\nimport time\n\nfrom inter.CheckRandCodeAnsyn import checkRandCodeAnsyn\nfrom inter.GetPassen"
},
{
"path": "inter/ConfirmSingleForQueueAsys.py",
"chars": 2464,
"preview": "# coding=utf-8\nimport json\nimport urllib\nfrom collections import OrderedDict\n\nfrom inter.QueryOrderWaitTime import query"
},
{
"path": "inter/GetPassCodeNewOrderAndLogin.py",
"chars": 2552,
"preview": "# coding=utf-8\nimport base64\nimport copy\nimport random\n\n\ndef getPassCodeNewOrderAndLogin(session, imgType):\n \"\"\"\n "
},
{
"path": "inter/GetPassengerDTOs.py",
"chars": 5655,
"preview": "# coding=utf-8\nimport json\n\nfrom config.TicketEnmu import ticket\nfrom myException.PassengerUserException import Passenge"
},
{
"path": "inter/GetQueueCount.py",
"chars": 6993,
"preview": "# coding=utf-8\nimport datetime\nimport sys\nimport time\nfrom collections import OrderedDict\nimport wrapcache\n\nimport Ticke"
},
{
"path": "inter/GetQueueCountAsync.py",
"chars": 5336,
"preview": "import TickerConfig\n\n[]# coding=utf-8\nimport datetime\nimport sys\nimport time\nfrom collections import OrderedDict\n\nimport"
},
{
"path": "inter/GetRandCode.py",
"chars": 2897,
"preview": "# coding=utf-8\nfrom PIL import Image\n\nfrom config.urlConf import urls\nfrom myUrllib.httpUtils import HTTPClient\nfrom ver"
},
{
"path": "inter/GetRepeatSubmitToken.py",
"chars": 1346,
"preview": "# coding=utf-8\nimport json\nimport re\n\n\nclass getRepeatSubmitToken:\n def __init__(self, session):\n self.session"
},
{
"path": "inter/GetSuccessRate.py",
"chars": 1634,
"preview": "from collections import OrderedDict\n\n\nfrom config.urlConf import urls\nimport TickerConfig\nfrom inter.SubmitOrderRequest "
},
{
"path": "inter/LiftTicketInit.py",
"chars": 602,
"preview": "# coding=utf-8\nimport re\n\n\nclass liftTicketInit:\n def __init__(self, session):\n self.session = session\n\n de"
},
{
"path": "inter/LoginAysnSuggest.py",
"chars": 701,
"preview": "# coding=utf-8\nfrom config.urlConf import urls\n\n\ndef loginAysnSuggest(session, username, password):\n \"\"\"\n 登录接口\n "
},
{
"path": "inter/LoginConf.py",
"chars": 472,
"preview": "# coding=utf-8\nfrom config.urlConf import urls\n\n\ndef loginConf(session):\n \"\"\"\n 判断登录是否需要验证码\n :param session:\n "
},
{
"path": "inter/PassengerInitApi.py",
"chars": 1295,
"preview": "import datetime\nimport wrapcache\nimport TickerConfig\nfrom config.urlConf import urls\nfrom inter.ConfirmHB import confirm"
},
{
"path": "inter/Query.py",
"chars": 10661,
"preview": "# coding=utf-8\nimport copy\nimport random\nimport wrapcache\nfrom config import urlConf\nfrom config.TicketEnmu import ticke"
},
{
"path": "inter/QueryOrderWaitTime.py",
"chars": 5271,
"preview": "# coding=utf-8\nimport copy\nimport time\n\nfrom config.TicketEnmu import ticket\nfrom config.emailConf import sendEmail\nfrom"
},
{
"path": "inter/SubmitOrderRequest.py",
"chars": 4354,
"preview": "# coding=utf-8\nimport datetime\nimport urllib\nfrom collections import OrderedDict\nimport TickerConfig\nfrom config.urlConf"
},
{
"path": "inter/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "myException/PassengerUserException.py",
"chars": 49,
"preview": "class PassengerUserException(Exception):\n pass"
},
{
"path": "myException/UserPasswordException.py",
"chars": 48,
"preview": "class UserPasswordException(Exception):\n pass"
},
{
"path": "myException/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "myException/balanceException.py",
"chars": 43,
"preview": "class balanceException(Exception):\n pass"
},
{
"path": "myException/ticketConfigException.py",
"chars": 48,
"preview": "class ticketConfigException(Exception):\n pass"
},
{
"path": "myException/ticketIsExitsException.py",
"chars": 49,
"preview": "class ticketIsExitsException(Exception):\n pass"
},
{
"path": "myException/ticketNumOutException.py",
"chars": 48,
"preview": "class ticketNumOutException(Exception):\n pass"
},
{
"path": "myUrllib/MySocketUtils.py",
"chars": 8344,
"preview": "# coding=utf-8\nimport json\nimport socket\nimport re\n# s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)\n# s.connect(("
},
{
"path": "myUrllib/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "myUrllib/httpUtils.py",
"chars": 7099,
"preview": "# -*- coding: utf8 -*-\nimport json\nimport random\nimport socket\nfrom collections import OrderedDict\nfrom time import slee"
},
{
"path": "requirements-docker37.txt",
"chars": 105,
"preview": "bs4==0.0.1\nrequests==2.18.4\nPillow\nwrapcache==1.0.8\nntplib==0.3.3\nselenium==3.11.0\nfake-useragent==0.1.11"
},
{
"path": "requirements.txt",
"chars": 226,
"preview": "beautifulsoup4==4.5.3\nbs4==0.0.1\nrequests==2.18.4\nPillow\nwrapcache==1.0.8\nntplib==0.3.3\nsklearn\nopencv-python\nkeras==2.2"
},
{
"path": "run.py",
"chars": 810,
"preview": "# -*- coding=utf-8 -*-\nimport argparse\nimport sys\n\n\ndef parser_arguments(argv):\n \"\"\"\n 不应该在这里定义,先放在这里\n :param ar"
},
{
"path": "station_name.txt",
"chars": 89814,
"preview": "var station_names ='@bjb|北京北|VAP|beijingbei|bjb|0@bjd|北京东|BOP|beijingdong|bjd|1@bji|北京|BJP|beijing|bj|2@bjn|北京南|VNP|beij"
},
{
"path": "tmp/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "tmp/log/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "verify/__init__.py",
"chars": 0,
"preview": ""
},
{
"path": "verify/localVerifyCode.py",
"chars": 3505,
"preview": "# coding: utf-8\nimport TickerConfig\n\nif TickerConfig.AUTO_CODE_TYPE == 2:\n import base64\n import os\n import cv2"
},
{
"path": "verify/mlearn_for_image.py",
"chars": 3032,
"preview": "# coding: utf-8\r\nimport TickerConfig\r\nif TickerConfig.AUTO_CODE_TYPE == 2:\r\n import sys\r\n\r\n import cv2\r\n import"
},
{
"path": "verify/pretreatment.py",
"chars": 2306,
"preview": "#! env python\n# coding: utf-8\n# 功能:对图像进行预处理,将文字部分单独提取出来\n# 并存放到ocr目录下\n# 文件名为原验证码文件的文件名\nimport TickerConfig\nif TickerConfi"
}
]
// ... and 2 more files (download for full content)
About this extraction
This page contains the full source code of the testerSunshine/12306 GitHub repository, extracted and formatted as plain text for AI agents and large language models (LLMs). The extraction includes 79 files (57.7 MB), approximately 138.8k tokens, and a symbol index with 209 extracted functions, classes, methods, constants, and types. Use this with OpenClaw, Claude, ChatGPT, Cursor, Windsurf, or any other AI tool that accepts text input. You can copy the full output to your clipboard or download it as a .txt file.
Extracted by GitExtract — free GitHub repo to text converter for AI. Built by Nikandr Surkov.