base on 🚂 12306 购票助手,支持集群,多账号,多任务购票以及 Web 页面管理 # 🚂 py12306 购票助手 分布式,多账号,多任务购票 ## Features - [x] 多日期查询余票 - [x] 自动打码下单 - [x] 用户状态恢复 - [x] 电话语音通知 - [x] 多账号、多任务、多线程支持 - [x] 单个任务多站点查询 - [x] 分布式运行 - [x] Docker 支持 - [x] 动态修改配置文件 - [x] 邮件通知 - [x] Web 管理页面 - [x] 微信消息通知 - [ ] 代理池支持 ([pyproxy-async](https://github.com/pjialin/pyproxy-async)) ## 使用 py12306 需要运行在 python 3.6 以上版本(其它版本暂未测试) **1. 安装依赖** ```bash git clone https://github.com/pjialin/py12306 pip install -r requirements.txt ``` **2. 配置程序** ```bash cp env.py.example env.py ``` 自动打码 (若快已停止服务,目前只能设置**free**打码模式) free 已对接到打码共享平台,[https://py12306-helper.pjialin.com](https://py12306-helper.pjialin.com/),欢迎参与分享 语音通知 语音验证码使用的是阿里云 API 市场上的一个服务商,需要到 [https://market.aliyun.com/products/56928004/cmapi026600.html](https://market.aliyun.com/products/56928004/cmapi026600.html) 购买后将 appcode 填写到配置中 **3. 启动前测试** 目前提供了一些简单的测试,包括用户账号检测,乘客信息检测,车站检测等 开始测试 -t ```bash python main.py -t ``` 测试通知消息 (语音, 邮件) -t -n ```bash # 默认不会进行通知测试,要对通知进行测试需要加上 -n 参数 python main.py -t -n ``` **4. 运行程序** ```bash python main.py ``` ### 参数列表 - -t 测试配置信息 - -t -n 测试配置信息以及通知消息 - -c 指定自定义配置文件位置 ### 分布式集群 集群依赖于 redis,目前支持情况 - 单台主节点多个子节点同时运行 - 主节点宕机后自动切换提升子节点为主节点 - 主节点恢复后自动恢复为真实主节点 - 配置通过主节点同步到所有子节点 - 主节点配置修改后无需重启子节点,支持自动更新 - 子节点消息实时同步到主节点 **使用** 将配置文件的中 `CLUSTER_ENABLED` 打开即开启分布式 目前提供了一个单独的子节点配置文件 `env.slave.py.example` 将文件修改为 `env.slave.py`, 通过 `python main.py -c env.slave.py` 即可快速启动 ## Docker 使用 **1. 将配置文件下载到本地** ```bash docker run --rm pjialin/py12306 cat /config/env.py > env.py # 或 curl https://raw.githubusercontent.com/pjialin/py12306/master/env.docker.py.example -o env.py ``` **2. 修改好配置后运行** ```bash docker run --rm --name py12306 -p 8008:8008 -d -v $(pwd):/config -v py12306:/data pjialin/py12306 ``` 当前目录会多一个 12306.log 的日志文件, `tail -f 12306.log` ### Docker-compose 中使用 **1. 复制配置文件** ``` cp docker-compose.yml.example docker-compose.yml ``` **2. 从 docker-compose 运行** 在`docker-compose.yml`所在的目录使用命令 ``` docker-compose up -d ``` ## Web 管理页面 目前支持用户和任务以及实时日志查看,更多功能后续会不断加入 **使用** 打开 Web 功能需要将配置中的 `WEB_ENABLE` 打开,启动程序后访问当前主机地址 + 端口号 (默认 8008) 即可,如 http://127.0.0.1:8008 ## 更新 - 19-01-10 - 支持分布式集群 - 19-01-11 - 配置文件支持动态修改 - 19-01-12 - 新增免费打码 - 19-01-14 - 新增 Web 页面支持 - 19-01-15 - 新增 钉钉通知 - 新增 Telegram 通知 - 新增 ServerChan 和 PushBear 微信推送 - 19-01-18 - 新增 CDN 查询 ## 截图 ### Web 管理页面 ![Web 管理页面图片](https://github.com/pjialin/py12306/blob/master/data/images/web.png) ### 下单成功 ![下单成功图片](https://github.com/pjialin/py12306/blob/master/data/images/order_success.png) ### 关于防封 目前查询和登录操作是分开的,查询是不依赖用户是否登录,放在 A 云 T 云容易被限制 ip,建议在其它网络环境下运行 QQ 交流群 [780289875](https://jq.qq.com/?_wv=1027&k=5PgzDwV),TG 群 [Py12306 交流](https://t.me/joinchat/F3sSegrF3x8KAmsd1mTu7w) ### Online IDE [![在 Gitpod 中打开](https://gitpod.io/button/open-in-gitpod.svg)](https://gitpod.io#https://github.com/pjialin/py12306) ## Thanks - 感谢大佬 [testerSunshine](https://github.com/testerSunshine/12306),借鉴了部分实现 - 感谢所有提供 pr 的大佬 - 感谢大佬 [zhaipro](https://github.com/zhaipro/easy12306) 的验证码本地识别模型与算法 ## License [Apache License.](https://github.com/pjialin/py12306/blob/master/LICENSE) ", Assign "at most 3 tags" to the expected json: {"id":"1594","tags":[]} "only from the tags list I provide: [{"id":77,"name":"3d"},{"id":89,"name":"agent"},{"id":17,"name":"ai"},{"id":54,"name":"algorithm"},{"id":24,"name":"api"},{"id":44,"name":"authentication"},{"id":3,"name":"aws"},{"id":27,"name":"backend"},{"id":60,"name":"benchmark"},{"id":72,"name":"best-practices"},{"id":39,"name":"bitcoin"},{"id":37,"name":"blockchain"},{"id":1,"name":"blog"},{"id":45,"name":"bundler"},{"id":58,"name":"cache"},{"id":21,"name":"chat"},{"id":49,"name":"cicd"},{"id":4,"name":"cli"},{"id":64,"name":"cloud-native"},{"id":48,"name":"cms"},{"id":61,"name":"compiler"},{"id":68,"name":"containerization"},{"id":92,"name":"crm"},{"id":34,"name":"data"},{"id":47,"name":"database"},{"id":8,"name":"declarative-gui "},{"id":9,"name":"deploy-tool"},{"id":53,"name":"desktop-app"},{"id":6,"name":"dev-exp-lib"},{"id":59,"name":"dev-tool"},{"id":13,"name":"ecommerce"},{"id":26,"name":"editor"},{"id":66,"name":"emulator"},{"id":62,"name":"filesystem"},{"id":80,"name":"finance"},{"id":15,"name":"firmware"},{"id":73,"name":"for-fun"},{"id":2,"name":"framework"},{"id":11,"name":"frontend"},{"id":22,"name":"game"},{"id":81,"name":"game-engine "},{"id":23,"name":"graphql"},{"id":84,"name":"gui"},{"id":91,"name":"http"},{"id":5,"name":"http-client"},{"id":51,"name":"iac"},{"id":30,"name":"ide"},{"id":78,"name":"iot"},{"id":40,"name":"json"},{"id":83,"name":"julian"},{"id":38,"name":"k8s"},{"id":31,"name":"language"},{"id":10,"name":"learning-resource"},{"id":33,"name":"lib"},{"id":41,"name":"linter"},{"id":28,"name":"lms"},{"id":16,"name":"logging"},{"id":76,"name":"low-code"},{"id":90,"name":"message-queue"},{"id":42,"name":"mobile-app"},{"id":18,"name":"monitoring"},{"id":36,"name":"networking"},{"id":7,"name":"node-version"},{"id":55,"name":"nosql"},{"id":57,"name":"observability"},{"id":46,"name":"orm"},{"id":52,"name":"os"},{"id":14,"name":"parser"},{"id":74,"name":"react"},{"id":82,"name":"real-time"},{"id":56,"name":"robot"},{"id":65,"name":"runtime"},{"id":32,"name":"sdk"},{"id":71,"name":"search"},{"id":63,"name":"secrets"},{"id":25,"name":"security"},{"id":85,"name":"server"},{"id":86,"name":"serverless"},{"id":70,"name":"storage"},{"id":75,"name":"system-design"},{"id":79,"name":"terminal"},{"id":29,"name":"testing"},{"id":12,"name":"ui"},{"id":50,"name":"ux"},{"id":88,"name":"video"},{"id":20,"name":"web-app"},{"id":35,"name":"web-server"},{"id":43,"name":"webassembly"},{"id":69,"name":"workflow"},{"id":87,"name":"yaml"}]" returns me the "expected json"