Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 0d7400734b | |||
| 48a6447c24 | |||
| 12ad5dd9e0 | |||
| b06eeddccc | |||
| 440cd5812b | |||
| 55dcfc1b3e | |||
| 4a8032665f | |||
| 6536c7fa9d | |||
| 2cd02453ec | |||
| 140e49342c | |||
| 93bbfe6029 | |||
| 6b1424b1c4 | |||
| efb5ed481e | |||
| e54a221f34 |
+135
-11
@@ -122,15 +122,26 @@ python scripts/agent_poller.py --action get --issue N
|
|||||||
|
|
||||||
### 3. 开发 / 修复
|
### 3. 开发 / 修复
|
||||||
|
|
||||||
|
**第零步:判断修复类型。** 不同修复类型走不同验证路径,**必须在开发前确认**:
|
||||||
|
|
||||||
|
| 类型 | 特征 | 示例 | 验证方式 |
|
||||||
|
|------|------|------|----------|
|
||||||
|
| **代码级修复** | 确定性逻辑错误、字段缺失、类型不对 | null check、type 标准化、字段补齐 | UT + pytest |
|
||||||
|
| **质量级修复** | 涉及 LLM 输出质量、覆盖率、语义判断 | Layer C audit、覆盖率提升、prompt 优化 | **必须 pipeline + e2e** |
|
||||||
|
|
||||||
|
**质量级修复必须在步骤 5-6 中实际运行 pipeline 并确认 Layer A+B+C 全部通过。**
|
||||||
|
如果无法运行 pipeline(API 不可用等),**禁止关闭 Issue** — 在 PR 和 Issue 中标注 `⚠ 待 e2e 验证`,保持 Issue open 等待 verifier 执行。
|
||||||
|
|
||||||
```
|
```
|
||||||
1. git pull origin main
|
1. [判定] 是代码级修复还是质量级修复?
|
||||||
2. git checkout -b dev/issue-N-<slug>
|
2. git pull origin main
|
||||||
3. 修改功能代码 + 更新/补充 UT 和接口集成测试
|
3. git checkout -b dev/issue-N-<slug>
|
||||||
4. python -m pytest -v # 本地全量 UT/集成测试
|
4. 修改功能代码 + 更新/补充 UT 和接口集成测试
|
||||||
5. python scripts/run_pipeline.py --input "input/<文档>.docx" # 运行完整 pipeline
|
5. python -m pytest -v # 本地全量 UT/集成测试
|
||||||
6. python -m pytest tests/acceptance/ -v --run-acceptance # e2e 验收 (Layer A+B+C)
|
6. [仅质量级修复] python scripts/run_pipeline.py --input "input/<文档>.docx"
|
||||||
7. git commit -m "fix: <描述> - Closes #N"
|
7. [仅质量级修复] python -m pytest tests/acceptance/ -v --run-acceptance
|
||||||
8. git push origin dev/issue-N-<slug>
|
8. git commit -m "fix: <描述> - Closes #N"
|
||||||
|
9. git push origin dev/issue-N-<slug>
|
||||||
```
|
```
|
||||||
|
|
||||||
**开发原则:**
|
**开发原则:**
|
||||||
@@ -138,8 +149,21 @@ python scripts/agent_poller.py --action get --issue N
|
|||||||
- 新增功能必须有对应的测试覆盖
|
- 新增功能必须有对应的测试覆盖
|
||||||
- 关注 IR 一致性:对同一输入的多次运行结果应尽量稳定
|
- 关注 IR 一致性:对同一输入的多次运行结果应尽量稳定
|
||||||
- 关注功能覆盖率:确保 IR 覆盖了输入文档中的功能点
|
- 关注功能覆盖率:确保 IR 覆盖了输入文档中的功能点
|
||||||
- **验证是实际功能验证,不是 dry-run**:`pytest` 通过只是门槛,必须用真实输入文档实际运行 pipeline 确认功能生效
|
- **代码级修复**:UT 通过即可关闭 Issue
|
||||||
- **PR 前必须通过 e2e 验收 (Layer A+B+C)**:防止修复引入回归。若无法运行完整 pipeline(API 不可用等),至少在 PR 描述中注明
|
- **质量级修复**:必须 pipeline + e2e 全部通过才能关闭 Issue。无法运行 pipeline 时,PR 和 Issue 标注 `⚠ 待 e2e 验证`,**Issue 保持 open**
|
||||||
|
|
||||||
|
**质量级修复批处理策略:**
|
||||||
|
|
||||||
|
e2e 测试耗时且消耗大量 LLM token。对于质量级修复(Layer C audit、覆盖率、prompt 优化),**单个小改动看不出效果** — 只有 pytest 是无效测试。
|
||||||
|
|
||||||
|
| 策略 | 说明 |
|
||||||
|
|------|------|
|
||||||
|
| **批量改动** | 将同一方向的质量级 Issue(如多个 Layer C 问题)合并到一个分支,打包测试 |
|
||||||
|
| **集中验证** | 一批改动只跑一次 pipeline + e2e,避免每个小 PR 重复消耗 token |
|
||||||
|
| **改动-测试成本匹配** | 跑一次完整 e2e 的 token 成本值得对应多个相关改动的验证 |
|
||||||
|
| **禁止逐个微调** | 不允许对同一个质量 Issue 反复做单行改动 → 跑 pytest → 关 Issue → 被重开 的循环 |
|
||||||
|
|
||||||
|
**质量级修复闭环:** 分析 → 打包相关 Issue → 合并在一个分支改动 → 跑一次 pipeline + e2e → Layer A+B+C 全部通过 → 关 Issue
|
||||||
|
|
||||||
### 4. 提交 PR
|
### 4. 提交 PR
|
||||||
|
|
||||||
@@ -151,9 +175,15 @@ python scripts/agent_poller.py --action create-pr \
|
|||||||
--body "## Summary
|
--body "## Summary
|
||||||
- <改动摘要>
|
- <改动摘要>
|
||||||
|
|
||||||
|
## 修复类型
|
||||||
|
- [ ] 代码级修复(UT 可验证)
|
||||||
|
- [ ] 质量级修复(需 pipeline + e2e 验证)
|
||||||
|
|
||||||
## Test
|
## Test
|
||||||
- [x] pytest 全量通过 (XX passed, Y skipped)
|
- [x] pytest 全量通过 (XX passed, Y skipped)
|
||||||
- [x] UT / 集成测试已更新
|
- [x] UT / 集成测试已更新
|
||||||
|
- [ ] pipeline 运行通过(仅质量级修复)
|
||||||
|
- [ ] e2e 验收 Layer A+B+C 通过(仅质量级修复)
|
||||||
|
|
||||||
Closes #N"
|
Closes #N"
|
||||||
```
|
```
|
||||||
@@ -255,6 +285,48 @@ QE-Agent 开 Issue (qe-feedback / bug / ci-failure)
|
|||||||
--title "[test] issue 标题" --labels test-code --body "..."
|
--title "[test] issue 标题" --labels test-code --body "..."
|
||||||
```
|
```
|
||||||
- 多个 label 用逗号分隔,如 `--labels "ci-failure,product-code"`
|
- 多个 label 用逗号分隔,如 `--labels "ci-failure,product-code"`
|
||||||
|
- **研究调查 Issue** → `investigation` label(根因不明、需实验验证的探索性工作)
|
||||||
|
```bash
|
||||||
|
python scripts/agent_poller.py --action create-issue \
|
||||||
|
--title "[investigation] issue 标题" --labels investigation --body "..."
|
||||||
|
```
|
||||||
|
研究 Issue 的用途见下方"研究型修复流程"。
|
||||||
|
|
||||||
|
## 研究型修复流程
|
||||||
|
|
||||||
|
**当根因不明确时,禁止反复做小改动试错。** 必须走研究 → 确认 → 修复 的路径。
|
||||||
|
|
||||||
|
### 判断:我是在修复还是试探?
|
||||||
|
|
||||||
|
| 情况 | 行为 |
|
||||||
|
|------|------|
|
||||||
|
| 根因明确、修复方案确定 | 直接修复,走正常闭环 |
|
||||||
|
| 根因不明确、有多个可能原因 | **开研究 Issue** |
|
||||||
|
| 改动后不确定效果、想"试试看" | **开研究 Issue** |
|
||||||
|
|
||||||
|
### 研究 Issue 流程
|
||||||
|
|
||||||
|
```
|
||||||
|
原 Issue (product-code) ← blocked by ← 研究 Issue (investigation)
|
||||||
|
↓
|
||||||
|
跑 pipeline → 收集数据 → 对比分析
|
||||||
|
↓
|
||||||
|
确认根因 → 关闭研究 Issue → 修复原 Issue
|
||||||
|
```
|
||||||
|
|
||||||
|
具体步骤:
|
||||||
|
|
||||||
|
1. **创建研究 Issue**:`--labels investigation`,描述要验证的假设和实验方法
|
||||||
|
2. **阻断原 Issue**:研究 Issue 创建后,在原 Issue 评论"阻塞: #研究Issue"
|
||||||
|
3. **实验验证**:在研究分支上跑 pipeline,收集 Layer A/B/C 数据,对比基线
|
||||||
|
4. **得出结论**:在研究 Issue 中记录实验结果和根因确认
|
||||||
|
5. **修复原 Issue**:确认根因后,在原 Issue 分支上实施修复
|
||||||
|
6. **关闭研究 Issue**:根因确认,修复完成,关闭研究 Issue
|
||||||
|
|
||||||
|
### 关键原则
|
||||||
|
- 一次研究 Issue 可以对应多个原 Issue(同一根因导致的多个症状)
|
||||||
|
- 研究 Issue 也遵循正常的 PR + CI 流程(但可以包含调试代码、日志等)
|
||||||
|
- 不确定的改动宁可开研究 Issue,也不要直接关原 Issue
|
||||||
|
|
||||||
## agent_poller 命令速查
|
## agent_poller 命令速查
|
||||||
|
|
||||||
@@ -287,9 +359,61 @@ QE-Agent 开 Issue (qe-feedback / bug / ci-failure)
|
|||||||
- [ ] **CI**:`agent_poller.py --action pr-status` 确认 CI 通过
|
- [ ] **CI**:`agent_poller.py --action pr-status` 确认 CI 通过
|
||||||
- [ ] **合并**:`agent_poller.py --action merge-pr` 合并 PR
|
- [ ] **合并**:`agent_poller.py --action merge-pr` 合并 PR
|
||||||
- [ ] **验证**:用真实输入文档实际运行 pipeline,确认功能生效(非 dry-run)
|
- [ ] **验证**:用真实输入文档实际运行 pipeline,确认功能生效(非 dry-run)
|
||||||
- [ ] **关闭**:验证通过后 `--action close-issue`
|
- [ ] **关闭**:验证通过后 `--action close-issue`(关闭 comment 必须符合下方"Issue 关闭规范")
|
||||||
- [ ] **复盘**:`agent_poller.py --action lifecycle` 确认全流程完成
|
- [ ] **复盘**:`agent_poller.py --action lifecycle` 确认全流程完成
|
||||||
|
|
||||||
|
## Issue 关闭规范
|
||||||
|
|
||||||
|
**关闭 Issue 时的 comment 必须包含以下四个要素,缺一不可:**
|
||||||
|
|
||||||
|
```
|
||||||
|
## 问题
|
||||||
|
<一句话描述 Issue 的症状>
|
||||||
|
|
||||||
|
## 根因
|
||||||
|
<明确指出导致问题的根本原因,不是表面现象>
|
||||||
|
|
||||||
|
## 修复
|
||||||
|
<这个改动如何消除根因?为什么这个方案是正确的?>
|
||||||
|
|
||||||
|
## 验证
|
||||||
|
<具体的验证步骤和结果,不是空泛的"已通过">
|
||||||
|
```
|
||||||
|
|
||||||
|
**禁止的关闭 comment:**
|
||||||
|
- "PR merged, 验证通过" — 没有说明根因和验证方式
|
||||||
|
- "自行验证通过,变更已合入 main" — 没有说明验证了什么
|
||||||
|
- 任何缺少上述四个要素的关闭 comment
|
||||||
|
|
||||||
|
**示例(正确):**
|
||||||
|
```
|
||||||
|
## 问题
|
||||||
|
_measure_coverage 将 0/0 维度 rate 算作 0%,拉低 overall 均值。
|
||||||
|
|
||||||
|
## 根因
|
||||||
|
`0 / max(0, 1) = 0%`,diagram 维度无内容时 rate 为 0% 并参与均分。
|
||||||
|
|
||||||
|
## 修复
|
||||||
|
引入 _safe_rate():total=0 时 rate=1.0。overall 均分排除 total=0 的维度。
|
||||||
|
|
||||||
|
## 验证
|
||||||
|
- pytest: 102 passed, 13 skipped
|
||||||
|
- test_layer_b_coverage: PASSED, overall 57.4%→86.1%
|
||||||
|
- 命令行确认: Section 100% + Table 72.2% → Overall 86.1%
|
||||||
|
```
|
||||||
|
|
||||||
|
## 禁止模式
|
||||||
|
|
||||||
|
以下行为模式被明确禁止。发现自己在做以下任何一件事,立即停止:
|
||||||
|
|
||||||
|
| 禁止模式 | 为什么禁止 | 正确做法 |
|
||||||
|
|----------|-----------|----------|
|
||||||
|
| 单行改动 → 关 Issue → 重开 → 再改 的循环 | 说明根因没找到,在试错 | 开研究 Issue |
|
||||||
|
| 不跑 pipeline 就关质量级 Issue | 无法证明修复有效 | 跑 pipeline + e2e,或 Issue 保持 open |
|
||||||
|
| 关闭 comment 不写根因 | 无法判断修复是否正确 | 按 Issue 关闭规范写 |
|
||||||
|
| 对同一 Issue 连续提交 3 个以上 PR | 说明方向不对 | 暂停,开研究 Issue |
|
||||||
|
| pytest 绿了就关 Issue | pytest 只保证无回归,不保证功能正确 | 代码级可关,质量级必须 pipeline |
|
||||||
|
|
||||||
## Session 收尾
|
## Session 收尾
|
||||||
|
|
||||||
**当 session 即将结束时(用户要求结束、或完成当前轮询周期后准备退出),执行以下收尾动作:**
|
**当 session 即将结束时(用户要求结束、或完成当前轮询周期后准备退出),执行以下收尾动作:**
|
||||||
|
|||||||
@@ -86,7 +86,8 @@ COVERAGE_TARGET = float(os.environ.get("IR_COVERAGE_TARGET", "0.95"))
|
|||||||
ENSEMBLE_TEMPERATURES = [
|
ENSEMBLE_TEMPERATURES = [
|
||||||
float(os.environ.get("IR_ENSEMBLE_T1", "0.0")),
|
float(os.environ.get("IR_ENSEMBLE_T1", "0.0")),
|
||||||
float(os.environ.get("IR_ENSEMBLE_T2", "0.3")),
|
float(os.environ.get("IR_ENSEMBLE_T2", "0.3")),
|
||||||
float(os.environ.get("IR_ENSEMBLE_T3", "0.7")),
|
float(os.environ.get("IR_ENSEMBLE_T3", "0.5")),
|
||||||
|
float(os.environ.get("IR_ENSEMBLE_T4", "0.7")),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -186,6 +186,8 @@
|
|||||||
|
|
||||||
8. **开关关闭状态**:开关关闭时所有限制失效,这也必须作为一条规则输出(path: ["...", "开关关闭", "无限制"])。
|
8. **开关关闭状态**:开关关闭时所有限制失效,这也必须作为一条规则输出(path: ["...", "开关关闭", "无限制"])。
|
||||||
|
|
||||||
|
9. **功能完整性要求(重要)**:上下文包中的每个表格行、每条文字描述、每个逻辑树路径都必须被至少一条规则覆盖。仔细检查上下文包,确保不遗漏任何数据来源。如果上下文包中有表格,每条表格行至少生成一条对应规则。
|
||||||
|
|
||||||
{format_feedback}
|
{format_feedback}
|
||||||
|
|
||||||
## 输出格式
|
## 输出格式
|
||||||
|
|||||||
@@ -880,9 +880,9 @@ def run_ensemble_semantic_index(doc: dict) -> dict:
|
|||||||
if v:
|
if v:
|
||||||
print(f" {k}: {len(v)} 个问题")
|
print(f" {k}: {len(v)} 个问题")
|
||||||
|
|
||||||
# Feedback retry: re-run with coverage feedback (up to 2 retries, quality-gated)
|
# Feedback retry: re-run with coverage feedback (up to 3 retries, quality-gated)
|
||||||
retry_count = 0
|
retry_count = 0
|
||||||
while retry_count < 2:
|
while retry_count < 3:
|
||||||
feedback = _build_coverage_feedback(gaps)
|
feedback = _build_coverage_feedback(gaps)
|
||||||
if not feedback:
|
if not feedback:
|
||||||
break
|
break
|
||||||
@@ -906,13 +906,16 @@ def run_ensemble_semantic_index(doc: dict) -> dict:
|
|||||||
if src.get("section"):
|
if src.get("section"):
|
||||||
retry_sections.add(src["section"])
|
retry_sections.add(src["section"])
|
||||||
print(f" 重试新增 sections: {sorted(retry_sections)}", flush=True)
|
print(f" 重试新增 sections: {sorted(retry_sections)}", flush=True)
|
||||||
# Quality gate: only include retry if it improves coverage
|
# Quality gate: include retry if it adds new sections or doesn't regress coverage
|
||||||
trial_indices = semantic_indices + [retry_result]
|
trial_indices = semantic_indices + [retry_result]
|
||||||
trial_merged = ensemble_merge(trial_indices)
|
trial_merged = ensemble_merge(trial_indices)
|
||||||
trial_passed, trial_gaps = _quick_validate(trial_merged, doc, all_paths)
|
trial_passed, trial_gaps = _quick_validate(trial_merged, doc, all_paths)
|
||||||
trial_warnings = len(trial_gaps.get("coverage_warnings", []))
|
trial_warnings = len(trial_gaps.get("coverage_warnings", []))
|
||||||
trial_missing = len(trial_gaps.get("missing_table_rows", []))
|
trial_missing = len(trial_gaps.get("missing_table_rows", []))
|
||||||
if trial_warnings < pre_warnings or trial_missing < pre_missing_rows:
|
improved = trial_warnings < pre_warnings or trial_missing < pre_missing_rows
|
||||||
|
no_regression = trial_warnings <= pre_warnings and trial_missing <= pre_missing_rows
|
||||||
|
has_new_sections = len(retry_sections) > 0
|
||||||
|
if improved or (no_regression and has_new_sections):
|
||||||
semantic_indices.append(retry_result)
|
semantic_indices.append(retry_result)
|
||||||
merged = trial_merged
|
merged = trial_merged
|
||||||
passed, gaps = trial_passed, trial_gaps
|
passed, gaps = trial_passed, trial_gaps
|
||||||
|
|||||||
@@ -174,11 +174,25 @@ def _normalize_rule(rule: dict) -> dict:
|
|||||||
sources = rule.get("sources", [])
|
sources = rule.get("sources", [])
|
||||||
valid_types = {"table", "text", "logic_tree"}
|
valid_types = {"table", "text", "logic_tree"}
|
||||||
|
|
||||||
|
def _clean_section(val):
|
||||||
|
"""Normalize section value: list→first element, ensure string."""
|
||||||
|
if isinstance(val, list):
|
||||||
|
return str(val[0]).strip() if val else ""
|
||||||
|
if isinstance(val, str):
|
||||||
|
return val.strip()
|
||||||
|
return str(val).strip() if val else ""
|
||||||
|
|
||||||
|
# Normalize section fields that might be lists (LLM format instability)
|
||||||
|
for s in sources:
|
||||||
|
sec = s.get("section")
|
||||||
|
if sec is not None:
|
||||||
|
s["section"] = _clean_section(sec)
|
||||||
|
|
||||||
# try to infer a default section from the rule path
|
# try to infer a default section from the rule path
|
||||||
default_section = ""
|
default_section = ""
|
||||||
for s in sources:
|
for s in sources:
|
||||||
sec = s.get("section", "")
|
sec = s.get("section", "")
|
||||||
if sec and sec.strip():
|
if sec and isinstance(sec, str) and sec.strip():
|
||||||
default_section = sec.strip()
|
default_section = sec.strip()
|
||||||
break
|
break
|
||||||
if not default_section:
|
if not default_section:
|
||||||
@@ -192,7 +206,12 @@ def _normalize_rule(rule: dict) -> dict:
|
|||||||
if stype and stype not in valid_types:
|
if stype and stype not in valid_types:
|
||||||
src["type"] = "text"
|
src["type"] = "text"
|
||||||
stype = "text"
|
stype = "text"
|
||||||
if stype in ("table", "text"):
|
if stype == "table":
|
||||||
|
if not src.get("section"):
|
||||||
|
src["section"] = default_section
|
||||||
|
if src.get("row") is None:
|
||||||
|
src["row"] = 0
|
||||||
|
elif stype == "text":
|
||||||
if not src.get("section"):
|
if not src.get("section"):
|
||||||
src["section"] = default_section
|
src["section"] = default_section
|
||||||
else:
|
else:
|
||||||
|
|||||||
@@ -512,6 +512,18 @@ class TestNormalizeRule:
|
|||||||
normalized = _normalize_rule(rule)
|
normalized = _normalize_rule(rule)
|
||||||
assert "section" not in normalized["sources"][0]
|
assert "section" not in normalized["sources"][0]
|
||||||
|
|
||||||
|
|
||||||
|
def test_normalize_table_source_null_row(self):
|
||||||
|
"""Table source with null row gets row=0 (defensive)."""
|
||||||
|
rule = {
|
||||||
|
"trigger": {"conditions": [{"signal": "x", "operator": "==", "value": "1"}]},
|
||||||
|
"sources": [
|
||||||
|
{"type": "table", "section": "3.1 功能", "row": None},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
normalized = _normalize_rule(rule)
|
||||||
|
assert normalized["sources"][0]["row"] == 0
|
||||||
|
|
||||||
def test_normalize_source_invalid_type(self):
|
def test_normalize_source_invalid_type(self):
|
||||||
"""Invalid source types (LLM hallucinations) are normalized to text."""
|
"""Invalid source types (LLM hallucinations) are normalized to text."""
|
||||||
rule = {
|
rule = {
|
||||||
@@ -538,3 +550,28 @@ class TestNormalizeRule:
|
|||||||
assert len(normalized["sources"]) == 1
|
assert len(normalized["sources"]) == 1
|
||||||
assert normalized["sources"][0]["type"] == "text"
|
assert normalized["sources"][0]["type"] == "text"
|
||||||
assert normalized["sources"][0]["section"] == "3.1 策略"
|
assert normalized["sources"][0]["section"] == "3.1 策略"
|
||||||
|
|
||||||
|
def test_normalize_section_is_list(self):
|
||||||
|
"""Section field that is a list (LLM format bug) is normalized to string."""
|
||||||
|
rule = {
|
||||||
|
"trigger": {"conditions": [{"signal": "x", "operator": "==", "value": "1"}]},
|
||||||
|
"sources": [
|
||||||
|
{"type": "table", "section": ["状态", "系统设置"], "row": 1},
|
||||||
|
{"type": "text", "section": ["后台限制"], "text_snippet": "x"},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
normalized = _normalize_rule(rule)
|
||||||
|
assert normalized["sources"][0]["section"] == "状态"
|
||||||
|
assert normalized["sources"][1]["section"] == "后台限制"
|
||||||
|
|
||||||
|
def test_normalize_section_is_empty_list(self):
|
||||||
|
"""Empty list section falls back to rule path."""
|
||||||
|
rule = {
|
||||||
|
"trigger": {"conditions": [{"signal": "x", "operator": "==", "value": "1"}]},
|
||||||
|
"path": "4.2 关闭流程 > decision",
|
||||||
|
"sources": [
|
||||||
|
{"type": "table", "section": [], "row": 1},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
normalized = _normalize_rule(rule)
|
||||||
|
assert normalized["sources"][0]["section"] == "4.2 关闭流程"
|
||||||
|
|||||||
@@ -83,8 +83,8 @@ def test_output_dir_structure():
|
|||||||
|
|
||||||
|
|
||||||
def test_ensemble_temperatures_count():
|
def test_ensemble_temperatures_count():
|
||||||
"""Should have exactly 3 ensemble temperatures."""
|
"""Should have exactly 4 ensemble temperatures."""
|
||||||
assert len(config.ENSEMBLE_TEMPERATURES) == 3
|
assert len(config.ENSEMBLE_TEMPERATURES) == 4
|
||||||
|
|
||||||
|
|
||||||
def test_max_tokens_is_int():
|
def test_max_tokens_is_int():
|
||||||
|
|||||||
Reference in New Issue
Block a user