新增 README.md

master
Mufanc 3 years ago
parent b85fe66bcb
commit 4fcbcdc7cb

@ -0,0 +1,52 @@
## iSmart 课程自动姬 v1.0.0
> <div align="center"><b>「不止于自动化,追求极致效率」</b></div><br/>
>
> * 如果你觉得这个脚本好用,请点一个 Star⭐你的 Star 就是作者更新最大的动力
### 效果展示
* 拥有更好的题型适应性,理论上适配所有客观题种类
* 提升稳定性,中途宕机概率大大降低
* 采用全新思路,相较 [自动化方案](https://github.com/Mufanc/iSmartAuto) ,效率提升超过 1000%
![](images/demo.png)
### 工作原理
&emsp;&emsp;使用抓包工具分析客户端流量,可以得知 iSmart 客户端采用的判题方式为本地评判。也就是说会首先将题目和答案一同下载下来,完成答题后使用用户的计算机完成判分,最后将分数回传。这样一来就为爬取答案提供了可能,脚本会根据提供的用户名和密码完成登录,然后将习题的答案下载下来,为进一步地自动答题做好准备。
&emsp;&emsp;一次偶然的机会,我发现 iSmart 客户端其实就是一个套壳的 Chromium在其启动参数中加上 `--remote-debugging-port=9222` 参数后,其中页面便能够在 [chrome://inspect](chrome://inspect) 中被调试:
![](images/inspect.png)
&emsp;&emsp;进而,可以通过 Python 调用 [Chrome DevTools Protocol](https://chromedevtools.github.io/devtools-protocol/) cdp完成答案的自动提交
#### Q&A
* **Q** 既然是回传分数,那为何不用 Python 直接将分数上报,反而要走 cdp
> 上报分数的请求中有疑似 Hash 的字段 `ut`,且生成 `ut` 的方法 native无法通过分析 JavaScript 得到(有木有大佬会 ollydbg 的来交个 PR
<br/>
* **Q** 使用这个脚本,会不会被检测到作弊?
> 不排除这样的可能性,相较自动化而言,目前的方式提交的数据尚不完整(但成绩和学习时长会被记录),若是仔细比对,有可能会发现数据异常
### 使用方法
&emsp;&emsp;修改 `configs.yml` 中的账号和密码,保证与 iSmart 客户端中登录的账号一致,然后修改 iSmart 的启动快捷方式,增加参数 `--remote-debugging-port=9222`
![](images/edit-lnk.png)
&emsp;&emsp;此时运行 main.py启动 iSmart 客户端,进入某本书籍的教材学习页(如下图),脚本会自动提交成绩。
![](images/booklearn.png)
### 开发者的话
&emsp;&emsp;该项目尚处于起步阶段,项目结构还没有完全确定下来,所以后续可能会经历多次重构。目前很多功能虽然存在于源码中,但还不完善或者未经测试,可能造成意料之外的结果,所以在使用前还请三思

@ -40,9 +40,22 @@ class Browser(object):
for page in pages: for page in pages:
if re.match(r'.*me.ismartlearning.cn/center/student/course/bookLearn\.html.*', page['url']): if re.match(r'.*me.ismartlearning.cn/center/student/course/bookLearn\.html.*', page['url']):
return Page(page['url'], page['webSocketDebuggerUrl']) return Page(page['url'], page['webSocketDebuggerUrl'])
await asyncio.sleep(2) # 这样写跟套 finally 有区别
except httpx.ConnectError: except httpx.ConnectError:
await asyncio.sleep(2) pass
await asyncio.sleep(2)
async def any_http_page(self): # 等待任意 http 页面
async with httpx.AsyncClient() as client:
while True:
logger.info('等待可用页面...')
try:
pages = (await client.get(f'http://127.0.0.1:{self.port}/json')).json()
for page in pages:
if re.match(r'https?://.*', page['url']):
return Page(page['url'], page['webSocketDebuggerUrl'])
except httpx.ConnectError:
pass
await asyncio.sleep(2)
class Page(object): class Page(object):

@ -15,7 +15,7 @@ from .spider import Spider
random_args = { # 不同题型对应的随机时长和分数范围 random_args = { # 不同题型对应的随机时长和分数范围
'1': { # 单选题 '1': { # 单选题
'time': (20, 60), # 完成时长 / 秒 'time': (20, 60), # 完成时长 / 秒
'score': 1 # 得分 (归一化, 向上至满分) 'score': 1 # 得分 (归一化, 向上随机取至满分)
}, },
'2': { # 多选题 '2': { # 多选题
'time': (40, 120), 'time': (40, 120),
@ -48,7 +48,7 @@ random_args = { # 不同题型对应的随机时长和分数范围
'11': { # 选词填空 '11': { # 选词填空
'time': (30, 90), 'time': (30, 90),
'score': 0.9 'score': 0.9
}, }
} }

@ -1,13 +1,18 @@
# Todo: 每次 commit 之前务必清除账号密码 # Todo: 每次 commit 之前务必清除账号密码
# iSmart 客户端相关配置 # 刷课配置
project:
skip-finished: true # 跳过已完成任务(暂不支持)
# iSmart 客户端配置
browser: browser:
executable: Z:\iSmart\client\iSmart.exe # 客户端可执行文件的路径 executable: Z:\iSmart\client\iSmart.exe # 客户端可执行文件的路径
args: # 启动 iSmart 客户端时额外提供的参数 args: # 启动 iSmart 客户端时额外提供的参数
- --disable-web-security - --disable-web-security
port: 9222 # devTools 调试端口 port: 9222 # devTools 调试端口
# 用户相关配置(务必保持账号密码与 iSmart 中已登录的相同) # 用户配置(务必保持账号密码与 iSmart 中已登录的相同)
user: user:
username: <用户名> # 手机号 username: <用户名> # 手机号
password: <密码> # 密码 password: <密码> # 密码

Binary file not shown.

After

Width:  |  Height:  |  Size: 71 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 64 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 25 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Loading…
Cancel
Save