@jackwener/opencli 0.7.0 → 0.7.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +190 -28
- package/README.md +4 -4
- package/README.zh-CN.md +4 -4
- package/SKILL.md +15 -4
- package/dist/browser.js +2 -3
- package/dist/engine.js +2 -1
- package/dist/main.js +10 -2
- package/dist/output.js +2 -1
- package/dist/registry.d.ts +1 -8
- package/dist/snapshotFormatter.d.ts +9 -0
- package/dist/snapshotFormatter.js +352 -15
- package/dist/snapshotFormatter.test.d.ts +7 -0
- package/dist/snapshotFormatter.test.js +521 -0
- package/dist/validate.d.ts +14 -2
- package/dist/verify.d.ts +14 -2
- package/package.json +2 -2
- package/src/browser.ts +2 -4
- package/src/engine.ts +4 -1
- package/src/main.ts +10 -2
- package/src/output.ts +2 -1
- package/src/registry.ts +1 -8
- package/src/snapshotFormatter.test.ts +579 -0
- package/src/snapshotFormatter.ts +399 -13
- package/src/validate.ts +19 -4
- package/src/verify.ts +17 -3
- package/vitest.config.ts +15 -1
package/LICENSE
CHANGED
|
@@ -1,28 +1,190 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
1
|
+
Apache License
|
|
2
|
+
Version 2.0, January 2004
|
|
3
|
+
http://www.apache.org/licenses/
|
|
4
|
+
|
|
5
|
+
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
|
6
|
+
|
|
7
|
+
1. Definitions.
|
|
8
|
+
|
|
9
|
+
"License" shall mean the terms and conditions for use, reproduction,
|
|
10
|
+
and distribution as defined by Sections 1 through 9 of this document.
|
|
11
|
+
|
|
12
|
+
"Licensor" shall mean the copyright owner or entity authorized by
|
|
13
|
+
the copyright owner that is granting the License.
|
|
14
|
+
|
|
15
|
+
"Legal Entity" shall mean the union of the acting entity and all
|
|
16
|
+
other entities that control, are controlled by, or are under common
|
|
17
|
+
control with that entity. For the purposes of this definition,
|
|
18
|
+
"control" means (i) the power, direct or indirect, to cause the
|
|
19
|
+
direction or management of such entity, whether by contract or
|
|
20
|
+
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
|
21
|
+
outstanding shares, or (iii) beneficial ownership of such entity.
|
|
22
|
+
|
|
23
|
+
"You" (or "Your") shall mean an individual or Legal Entity
|
|
24
|
+
exercising permissions granted by this License.
|
|
25
|
+
|
|
26
|
+
"Source" form shall mean the preferred form for making modifications,
|
|
27
|
+
including but not limited to software source code, documentation
|
|
28
|
+
source, and configuration files.
|
|
29
|
+
|
|
30
|
+
"Object" form shall mean any form resulting from mechanical
|
|
31
|
+
transformation or translation of a Source form, including but
|
|
32
|
+
not limited to compiled object code, generated documentation,
|
|
33
|
+
and conversions to other media types.
|
|
34
|
+
|
|
35
|
+
"Work" shall mean the work of authorship, whether in Source or
|
|
36
|
+
Object form, made available under the License, as indicated by a
|
|
37
|
+
copyright notice that is included in or attached to the work
|
|
38
|
+
(an example is provided in the Appendix below).
|
|
39
|
+
|
|
40
|
+
"Derivative Works" shall mean any work, whether in Source or Object
|
|
41
|
+
form, that is based on (or derived from) the Work and for which the
|
|
42
|
+
editorial revisions, annotations, elaborations, or other modifications
|
|
43
|
+
represent, as a whole, an original work of authorship. For the purposes
|
|
44
|
+
of this License, Derivative Works shall not include works that remain
|
|
45
|
+
separable from, or merely link (or bind by name) to the interfaces of,
|
|
46
|
+
the Work and Derivative Works thereof.
|
|
47
|
+
|
|
48
|
+
"Contribution" shall mean any work of authorship, including
|
|
49
|
+
the original version of the Work and any modifications or additions
|
|
50
|
+
to that Work or Derivative Works thereof, that is intentionally
|
|
51
|
+
submitted to the Licensor for inclusion in the Work by the copyright owner
|
|
52
|
+
or by an individual or Legal Entity authorized to submit on behalf of
|
|
53
|
+
the copyright owner. For the purposes of this definition, "submitted"
|
|
54
|
+
means any form of electronic, verbal, or written communication sent
|
|
55
|
+
to the Licensor or its representatives, including but not limited to
|
|
56
|
+
communication on electronic mailing lists, source code control systems,
|
|
57
|
+
and issue tracking systems that are managed by, or on behalf of, the
|
|
58
|
+
Licensor for the purpose of discussing and improving the Work, but
|
|
59
|
+
excluding communication that is conspicuously marked or otherwise
|
|
60
|
+
designated in writing by the copyright owner as "Not a Contribution."
|
|
61
|
+
|
|
62
|
+
"Contributor" shall mean Licensor and any individual or Legal Entity
|
|
63
|
+
on behalf of whom a Contribution has been received by the Licensor and
|
|
64
|
+
subsequently incorporated within the Work.
|
|
65
|
+
|
|
66
|
+
2. Grant of Copyright License. Subject to the terms and conditions of
|
|
67
|
+
this License, each Contributor hereby grants to You a perpetual,
|
|
68
|
+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
69
|
+
copyright license to reproduce, prepare Derivative Works of,
|
|
70
|
+
publicly display, publicly perform, sublicense, and distribute the
|
|
71
|
+
Work and such Derivative Works in Source or Object form.
|
|
72
|
+
|
|
73
|
+
3. Grant of Patent License. Subject to the terms and conditions of
|
|
74
|
+
this License, each Contributor hereby grants to You a perpetual,
|
|
75
|
+
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
|
76
|
+
(except as stated in this section) patent license to make, have made,
|
|
77
|
+
use, offer to sell, sell, import, and otherwise transfer the Work,
|
|
78
|
+
where such license applies only to those patent claims licensable
|
|
79
|
+
by such Contributor that are necessarily infringed by their
|
|
80
|
+
Contribution(s) alone or by combination of their Contribution(s)
|
|
81
|
+
with the Work to which such Contribution(s) was submitted. If You
|
|
82
|
+
institute patent litigation against any entity (including a
|
|
83
|
+
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
|
84
|
+
or a Contribution incorporated within the Work constitutes direct
|
|
85
|
+
or contributory patent infringement, then any patent licenses
|
|
86
|
+
granted to You under this License for that Work shall terminate
|
|
87
|
+
as of the date such litigation is filed.
|
|
88
|
+
|
|
89
|
+
4. Redistribution. You may reproduce and distribute copies of the
|
|
90
|
+
Work or Derivative Works thereof in any medium, with or without
|
|
91
|
+
modifications, and in Source or Object form, provided that You
|
|
92
|
+
meet the following conditions:
|
|
93
|
+
|
|
94
|
+
(a) You must give any other recipients of the Work or
|
|
95
|
+
Derivative Works a copy of this License; and
|
|
96
|
+
|
|
97
|
+
(b) You must cause any modified files to carry prominent notices
|
|
98
|
+
stating that You changed the files; and
|
|
99
|
+
|
|
100
|
+
(c) You must retain, in the Source form of any Derivative Works
|
|
101
|
+
that You distribute, all copyright, patent, trademark, and
|
|
102
|
+
attribution notices from the Source form of the Work,
|
|
103
|
+
excluding those notices that do not pertain to any part of
|
|
104
|
+
the Derivative Works; and
|
|
105
|
+
|
|
106
|
+
(d) If the Work includes a "NOTICE" text file as part of its
|
|
107
|
+
distribution, then any Derivative Works that You distribute must
|
|
108
|
+
include a readable copy of the attribution notices contained
|
|
109
|
+
within such NOTICE file, excluding any notices that do not
|
|
110
|
+
pertain to any part of the Derivative Works, in at least one
|
|
111
|
+
of the following places: within a NOTICE text file distributed
|
|
112
|
+
as part of the Derivative Works; within the Source form or
|
|
113
|
+
documentation, if provided along with the Derivative Works; or,
|
|
114
|
+
within a display generated by the Derivative Works, if and
|
|
115
|
+
wherever such third-party notices normally appear. The contents
|
|
116
|
+
of the NOTICE file are for informational purposes only and
|
|
117
|
+
do not modify the License. You may add Your own attribution
|
|
118
|
+
notices within Derivative Works that You distribute, alongside
|
|
119
|
+
or as an addendum to the NOTICE text from the Work, provided
|
|
120
|
+
that such additional attribution notices cannot be construed
|
|
121
|
+
as modifying the License.
|
|
122
|
+
|
|
123
|
+
You may add Your own copyright statement to Your modifications and
|
|
124
|
+
may provide additional or different license terms and conditions
|
|
125
|
+
for use, reproduction, or distribution of Your modifications, or
|
|
126
|
+
for any such Derivative Works as a whole, provided Your use,
|
|
127
|
+
reproduction, and distribution of the Work otherwise complies with
|
|
128
|
+
the conditions stated in this License.
|
|
129
|
+
|
|
130
|
+
5. Submission of Contributions. Unless You explicitly state otherwise,
|
|
131
|
+
any Contribution intentionally submitted for inclusion in the Work
|
|
132
|
+
by You to the Licensor shall be under the terms and conditions of
|
|
133
|
+
this License, without any additional terms or conditions.
|
|
134
|
+
Notwithstanding the above, nothing herein shall supersede or modify
|
|
135
|
+
the terms of any separate license agreement you may have executed
|
|
136
|
+
with Licensor regarding such Contributions.
|
|
137
|
+
|
|
138
|
+
6. Trademarks. This License does not grant permission to use the trade
|
|
139
|
+
names, trademarks, service marks, or product names of the Licensor,
|
|
140
|
+
except as required for reasonable and customary use in describing the
|
|
141
|
+
origin of the Work and reproducing the content of the NOTICE file.
|
|
142
|
+
|
|
143
|
+
7. Disclaimer of Warranty. Unless required by applicable law or
|
|
144
|
+
agreed to in writing, Licensor provides the Work (and each
|
|
145
|
+
Contributor provides its Contributions) on an "AS IS" BASIS,
|
|
146
|
+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
147
|
+
implied, including, without limitation, any warranties or conditions
|
|
148
|
+
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
|
149
|
+
PARTICULAR PURPOSE. You are solely responsible for determining the
|
|
150
|
+
appropriateness of using or redistributing the Work and assume any
|
|
151
|
+
risks associated with Your exercise of permissions under this License.
|
|
152
|
+
|
|
153
|
+
8. Limitation of Liability. In no event and under no legal theory,
|
|
154
|
+
whether in tort (including negligence), contract, or otherwise,
|
|
155
|
+
unless required by applicable law (such as deliberate and grossly
|
|
156
|
+
negligent acts) or agreed to in writing, shall any Contributor be
|
|
157
|
+
liable to You for damages, including any direct, indirect, special,
|
|
158
|
+
incidental, or consequential damages of any character arising as a
|
|
159
|
+
result of this License or out of the use or inability to use the
|
|
160
|
+
Work (including but not limited to damages for loss of goodwill,
|
|
161
|
+
work stoppage, computer failure or malfunction, or any and all
|
|
162
|
+
other commercial damages or losses), even if such Contributor
|
|
163
|
+
has been advised of the possibility of such damages.
|
|
164
|
+
|
|
165
|
+
9. Accepting Warranty or Additional Liability. While redistributing
|
|
166
|
+
the Work or Derivative Works thereof, You may choose to offer,
|
|
167
|
+
and charge a fee for, acceptance of support, warranty, indemnity,
|
|
168
|
+
or other liability obligations and/or rights consistent with this
|
|
169
|
+
License. However, in accepting such obligations, You may act only
|
|
170
|
+
on Your own behalf and on Your sole responsibility, not on behalf
|
|
171
|
+
of any other Contributor, and only if You agree to indemnify,
|
|
172
|
+
defend, and hold each Contributor harmless for any liability
|
|
173
|
+
incurred by, or claims asserted against, such Contributor by reason
|
|
174
|
+
of your accepting any such warranty or additional liability.
|
|
175
|
+
|
|
176
|
+
END OF TERMS AND CONDITIONS
|
|
177
|
+
|
|
178
|
+
Copyright 2025 jackwener
|
|
179
|
+
|
|
180
|
+
Licensed under the Apache License, Version 2.0 (the "License");
|
|
181
|
+
you may not use this file except in compliance with the License.
|
|
182
|
+
You may obtain a copy of the License at
|
|
183
|
+
|
|
184
|
+
http://www.apache.org/licenses/LICENSE-2.0
|
|
185
|
+
|
|
186
|
+
Unless required by applicable law or agreed to in writing, software
|
|
187
|
+
distributed under the License is distributed on an "AS IS" BASIS,
|
|
188
|
+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
189
|
+
See the License for the specific language governing permissions and
|
|
190
|
+
limitations under the License.
|
package/README.md
CHANGED
|
@@ -134,10 +134,10 @@ npm install -g @jackwener/opencli@latest
|
|
|
134
134
|
|------|----------|------|
|
|
135
135
|
| **bilibili** | `hot` `search` `me` `favorite` `history` `feed` `subtitle` `dynamic` `ranking` `following` `user-videos` | 🔐 Browser |
|
|
136
136
|
| **zhihu** | `hot` `search` `question` | 🔐 Browser |
|
|
137
|
-
| **xiaohongshu** | `search` `notifications` `feed` `
|
|
137
|
+
| **xiaohongshu** | `search` `notifications` `feed` `user` | 🔐 Browser |
|
|
138
138
|
| **xueqiu** | `feed` `hot-stock` `hot` `search` `stock` `watchlist` | 🔐 Browser |
|
|
139
|
-
| **twitter** | `trending` `bookmarks` `profile` `search` `timeline` `following` `followers` `notifications` `post` `reply` `delete` `like` `article` `follow` `unfollow` `bookmark` `unbookmark` | 🔐 Browser |
|
|
140
|
-
| **reddit** | `hot` `frontpage` `search` `subreddit` | 🔐 Browser |
|
|
139
|
+
| **twitter** | `trending` `bookmarks` `profile` `search` `timeline` `thread` `following` `followers` `notifications` `post` `reply` `delete` `like` `article` `follow` `unfollow` `bookmark` `unbookmark` | 🔐 Browser |
|
|
140
|
+
| **reddit** | `hot` `frontpage` `popular` `search` `subreddit` `read` `user` `user-posts` `user-comments` `upvote` `save` `comment` `subscribe` `saved` `upvoted` | 🔐 Browser |
|
|
141
141
|
| **weibo** | `hot` | 🔐 Browser |
|
|
142
142
|
| **boss** | `search` `detail` | 🔐 Browser |
|
|
143
143
|
| **coupang** | `search` `add-to-cart` | 🔐 Browser |
|
|
@@ -232,4 +232,4 @@ The CI will automatically build, create a GitHub release, and publish to npm.
|
|
|
232
232
|
|
|
233
233
|
## License
|
|
234
234
|
|
|
235
|
-
[
|
|
235
|
+
[Apache-2.0](./LICENSE)
|
package/README.zh-CN.md
CHANGED
|
@@ -133,10 +133,10 @@ npm install -g @jackwener/opencli@latest
|
|
|
133
133
|
|------|------|------|
|
|
134
134
|
| **bilibili** | `hot` `search` `me` `favorite` `history` `feed` `subtitle` `dynamic` `ranking` `following` `user-videos` | 🔐 浏览器 |
|
|
135
135
|
| **zhihu** | `hot` `search` `question` | 🔐 浏览器 |
|
|
136
|
-
| **xiaohongshu** | `search` `notifications` `feed` `
|
|
136
|
+
| **xiaohongshu** | `search` `notifications` `feed` `user` | 🔐 浏览器 |
|
|
137
137
|
| **xueqiu** | `feed` `hot-stock` `hot` `search` `stock` `watchlist` | 🔐 浏览器 |
|
|
138
|
-
| **twitter** | `trending` `bookmarks` `profile` `search` `timeline` `following` `followers` `notifications` `post` `reply` `delete` `like` `article` `follow` `unfollow` `bookmark` `unbookmark` | 🔐 浏览器 |
|
|
139
|
-
| **reddit** | `hot` `frontpage` `search` `subreddit` | 🔐 浏览器 |
|
|
138
|
+
| **twitter** | `trending` `bookmarks` `profile` `search` `timeline` `thread` `following` `followers` `notifications` `post` `reply` `delete` `like` `article` `follow` `unfollow` `bookmark` `unbookmark` | 🔐 浏览器 |
|
|
139
|
+
| **reddit** | `hot` `frontpage` `popular` `search` `subreddit` `read` `user` `user-posts` `user-comments` `upvote` `save` `comment` `subscribe` `saved` `upvoted` | 🔐 浏览器 |
|
|
140
140
|
| **weibo** | `hot` | 🔐 浏览器 |
|
|
141
141
|
| **boss** | `search` `detail` | 🔐 浏览器 |
|
|
142
142
|
| **coupang** | `search` `add-to-cart` | 🔐 浏览器 |
|
|
@@ -213,4 +213,4 @@ git push --follow-tags
|
|
|
213
213
|
|
|
214
214
|
## License
|
|
215
215
|
|
|
216
|
-
[
|
|
216
|
+
[Apache-2.0](./LICENSE)
|
package/SKILL.md
CHANGED
|
@@ -68,7 +68,6 @@ opencli zhihu question --id 34816524 # 问题详情和回答
|
|
|
68
68
|
opencli xiaohongshu search --keyword "美食" # 搜索笔记
|
|
69
69
|
opencli xiaohongshu notifications # 通知(mentions/likes/connections)
|
|
70
70
|
opencli xiaohongshu feed --limit 10 # 推荐 Feed
|
|
71
|
-
opencli xiaohongshu me # 我的信息
|
|
72
71
|
opencli xiaohongshu user --uid xxx # 用户主页
|
|
73
72
|
|
|
74
73
|
# 雪球 Xueqiu (browser)
|
|
@@ -88,6 +87,7 @@ opencli twitter bookmarks --limit 20 # 获取收藏的书签推文
|
|
|
88
87
|
opencli twitter search --keyword "AI" # 搜索推文
|
|
89
88
|
opencli twitter profile elonmusk # 用户资料
|
|
90
89
|
opencli twitter timeline --limit 20 # 时间线
|
|
90
|
+
opencli twitter thread 1234567890 # 推文 thread(原文 + 回复)
|
|
91
91
|
opencli twitter article 1891511252174299446 # 推文长文内容
|
|
92
92
|
opencli twitter follow elonmusk # 关注用户
|
|
93
93
|
opencli twitter unfollow elonmusk # 取消关注
|
|
@@ -97,9 +97,20 @@ opencli twitter unbookmark https://x.com/... # 取消收藏
|
|
|
97
97
|
# Reddit (browser)
|
|
98
98
|
opencli reddit hot --limit 10 # 热门帖子
|
|
99
99
|
opencli reddit hot --subreddit programming # 指定子版块
|
|
100
|
-
opencli reddit frontpage --limit 10 # 首页
|
|
101
|
-
opencli reddit
|
|
102
|
-
opencli reddit
|
|
100
|
+
opencli reddit frontpage --limit 10 # 首页 /r/all
|
|
101
|
+
opencli reddit popular --limit 10 # /r/popular 热门
|
|
102
|
+
opencli reddit search --query "AI" --sort top --time week # 搜索(支持排序+时间过滤)
|
|
103
|
+
opencli reddit subreddit --name rust --sort top --time month # 子版块浏览(支持时间过滤)
|
|
104
|
+
opencli reddit read --post_id 1abc123 # 阅读帖子 + 评论
|
|
105
|
+
opencli reddit user --username spez # 用户资料(karma、注册时间)
|
|
106
|
+
opencli reddit user-posts --username spez # 用户发帖历史
|
|
107
|
+
opencli reddit user-comments --username spez # 用户评论历史
|
|
108
|
+
opencli reddit upvote --post_id xxx --direction up # 投票(up/down/none)
|
|
109
|
+
opencli reddit save --post_id xxx # 收藏帖子
|
|
110
|
+
opencli reddit comment --post_id xxx --text "Great!" # 发表评论
|
|
111
|
+
opencli reddit subscribe --subreddit python # 订阅子版块
|
|
112
|
+
opencli reddit saved --limit 10 # 我的收藏
|
|
113
|
+
opencli reddit upvoted --limit 10 # 我的赞
|
|
103
114
|
|
|
104
115
|
# V2EX (public + browser)
|
|
105
116
|
opencli v2ex hot --limit 10 # 热门话题
|
package/dist/browser.js
CHANGED
|
@@ -12,8 +12,7 @@ import { formatSnapshot } from './snapshotFormatter.js';
|
|
|
12
12
|
import { PKG_VERSION } from './version.js';
|
|
13
13
|
import { normalizeEvaluateSource } from './pipeline/template.js';
|
|
14
14
|
import { generateInterceptorJs, generateReadInterceptedJs } from './interceptor.js';
|
|
15
|
-
import { withTimeoutMs } from './runtime.js';
|
|
16
|
-
const CONNECT_TIMEOUT = parseInt(process.env.OPENCLI_BROWSER_CONNECT_TIMEOUT ?? '30', 10);
|
|
15
|
+
import { withTimeoutMs, DEFAULT_BROWSER_CONNECT_TIMEOUT } from './runtime.js';
|
|
17
16
|
const STDERR_BUFFER_LIMIT = 16 * 1024;
|
|
18
17
|
const INITIAL_TABS_TIMEOUT_MS = 1500;
|
|
19
18
|
const TAB_CLEANUP_TIMEOUT_MS = 2000;
|
|
@@ -299,7 +298,7 @@ export class PlaywrightMCP {
|
|
|
299
298
|
PlaywrightMCP._registerGlobalCleanup();
|
|
300
299
|
PlaywrightMCP._activeInsts.add(this);
|
|
301
300
|
this._state = 'connecting';
|
|
302
|
-
const timeout = opts.timeout ??
|
|
301
|
+
const timeout = opts.timeout ?? DEFAULT_BROWSER_CONNECT_TIMEOUT;
|
|
303
302
|
return new Promise((resolve, reject) => {
|
|
304
303
|
const isDebug = process.env.DEBUG?.includes('opencli:mcp');
|
|
305
304
|
const debugLog = (msg) => isDebug && console.error(`[opencli:mcp] ${msg}`);
|
package/dist/engine.js
CHANGED
|
@@ -100,7 +100,8 @@ async function discoverClisFromFs(dir) {
|
|
|
100
100
|
if (file.endsWith('.yaml') || file.endsWith('.yml')) {
|
|
101
101
|
registerYamlCli(filePath, site);
|
|
102
102
|
}
|
|
103
|
-
else if (file.endsWith('.js') && !file.endsWith('.d.js'))
|
|
103
|
+
else if ((file.endsWith('.js') && !file.endsWith('.d.js')) ||
|
|
104
|
+
(file.endsWith('.ts') && !file.endsWith('.d.ts') && !file.endsWith('.test.ts'))) {
|
|
104
105
|
promises.push(import(`file://${filePath}`).catch((err) => {
|
|
105
106
|
process.stderr.write(`Warning: failed to load module ${filePath}: ${err.message}\n`);
|
|
106
107
|
}));
|
package/dist/main.js
CHANGED
|
@@ -65,9 +65,17 @@ program.command('list').description('List all available CLI commands').option('-
|
|
|
65
65
|
console.log();
|
|
66
66
|
});
|
|
67
67
|
program.command('validate').description('Validate CLI definitions').argument('[target]', 'site or site/name')
|
|
68
|
-
.action(async (target) => {
|
|
68
|
+
.action(async (target) => {
|
|
69
|
+
const { validateClisWithTarget, renderValidationReport } = await import('./validate.js');
|
|
70
|
+
console.log(renderValidationReport(validateClisWithTarget([BUILTIN_CLIS, USER_CLIS], target)));
|
|
71
|
+
});
|
|
69
72
|
program.command('verify').description('Validate + smoke test').argument('[target]').option('--smoke', 'Run smoke tests', false)
|
|
70
|
-
.action(async (target, opts) => {
|
|
73
|
+
.action(async (target, opts) => {
|
|
74
|
+
const { verifyClis, renderVerifyReport } = await import('./verify.js');
|
|
75
|
+
const r = await verifyClis({ builtinClis: BUILTIN_CLIS, userClis: USER_CLIS, target, smoke: opts.smoke });
|
|
76
|
+
console.log(renderVerifyReport(r));
|
|
77
|
+
process.exitCode = r.ok ? 0 : 1;
|
|
78
|
+
});
|
|
71
79
|
program.command('explore').alias('probe').description('Explore a website: discover APIs, stores, and recommend strategies').argument('<url>').option('--site <name>').option('--goal <text>').option('--wait <s>', '', '3').option('--auto', 'Enable interactive fuzzing (simulate clicks to trigger lazy APIs)').option('--click <labels>', 'Comma-separated labels to click before fuzzing (e.g. "字幕,CC,评论")')
|
|
72
80
|
.action(async (url, opts) => { const { exploreUrl, renderExploreSummary } = await import('./explore.js'); const clickLabels = opts.click ? opts.click.split(',').map((s) => s.trim()) : undefined; console.log(renderExploreSummary(await exploreUrl(url, { BrowserFactory: PlaywrightMCP, site: opts.site, goal: opts.goal, waitSeconds: parseFloat(opts.wait), auto: opts.auto, clickLabels }))); });
|
|
73
81
|
program.command('synthesize').description('Synthesize CLIs from explore').argument('<target>').option('--top <n>', '', '3')
|
package/dist/output.js
CHANGED
|
@@ -85,7 +85,8 @@ function renderCsv(data, opts) {
|
|
|
85
85
|
for (const row of rows) {
|
|
86
86
|
console.log(columns.map(c => {
|
|
87
87
|
const v = String(row[c] ?? '');
|
|
88
|
-
return v.includes(',') || v.includes('"')
|
|
88
|
+
return v.includes(',') || v.includes('"') || v.includes('\n')
|
|
89
|
+
? `"${v.replace(/"/g, '""')}"` : v;
|
|
89
90
|
}).join(','));
|
|
90
91
|
}
|
|
91
92
|
}
|
package/dist/registry.d.ts
CHANGED
|
@@ -37,18 +37,11 @@ export interface InternalCliCommand extends CliCommand {
|
|
|
37
37
|
_lazy?: boolean;
|
|
38
38
|
_modulePath?: string;
|
|
39
39
|
}
|
|
40
|
-
export interface CliOptions {
|
|
40
|
+
export interface CliOptions extends Partial<Omit<CliCommand, 'args' | 'description'>> {
|
|
41
41
|
site: string;
|
|
42
42
|
name: string;
|
|
43
43
|
description?: string;
|
|
44
|
-
domain?: string;
|
|
45
|
-
strategy?: Strategy;
|
|
46
|
-
browser?: boolean;
|
|
47
44
|
args?: Arg[];
|
|
48
|
-
columns?: string[];
|
|
49
|
-
func?: (page: IPage, kwargs: Record<string, any>, debug?: boolean) => Promise<any>;
|
|
50
|
-
pipeline?: any[];
|
|
51
|
-
timeoutSeconds?: number;
|
|
52
45
|
}
|
|
53
46
|
export declare function cli(opts: CliOptions): CliCommand;
|
|
54
47
|
export declare function getRegistry(): Map<string, CliCommand>;
|
|
@@ -1,9 +1,18 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Aria snapshot formatter: parses Playwright MCP snapshot text into clean format.
|
|
3
|
+
*
|
|
4
|
+
* Multi-pass pipeline:
|
|
5
|
+
* 1. Parse & filter: strip annotations, metadata, noise roles, ads, decorators
|
|
6
|
+
* 2. Deduplicate: generic/text child matching parent label
|
|
7
|
+
* 3. Deduplicate: heading + link with identical labels
|
|
8
|
+
* 4. Deduplicate: nested identical links
|
|
9
|
+
* 5. Prune: empty containers (iterative bottom-up)
|
|
10
|
+
* 6. Collapse: single-child containers
|
|
3
11
|
*/
|
|
4
12
|
export interface FormatOptions {
|
|
5
13
|
interactive?: boolean;
|
|
6
14
|
compact?: boolean;
|
|
7
15
|
maxDepth?: number;
|
|
16
|
+
maxTextLength?: number;
|
|
8
17
|
}
|
|
9
18
|
export declare function formatSnapshot(raw: string, opts?: FormatOptions): string;
|