Files
hakorune/docs/development/current/main/phase245b-num_str-carrier-design.md

305 lines
7.3 KiB
Markdown
Raw Normal View History

# Phase 245B: num_str Carrier Design Document
**Status**: Design Phase
**Target**: `_parse_number` loop の `num_str` 文字列キャリア対応
**Scope**: Pattern 2 (loop with break) + 既存インフラ活用
---
## 1. num_str の役割
### 1.1 現在の `_parse_number` ループ構造
```nyash
fn _parse_number(s, p) {
local num_str = ""
local len = s.length()
loop(p < len) {
local ch = s.substring(p, p + 1)
if not _is_digit(ch) {
break
}
num_str = num_str + ch // ← StringAppend carrier update
p = p + 1
}
return num_str // or return { num_str, p }
}
```
### 1.2 num_str の役割定義
| 観点 | 決定 |
|------|------|
| **主目的** | digit を連結する文字列バッファ |
| **スコープ** | `_parse_number` 専用245B では) |
| **共有可能性** | 将来 `_atoi` / `_atof_loop` と共通化可能 |
| **初期値** | 空文字 `""` |
| **更新パターン** | Append のみ(削除・置換なし) |
### 1.3 将来の拡張候補245B では非対象)
- `_atoi`: `num_str``result: Integer` への変換ループ
- `_atof_loop`: 小数部・指数部を含む浮動小数点パース
- これらとの整合性は **Phase 246+** で検討
---
## 2. 許可する UpdateExpr パターン
### 2.1 最小セット245B 対象)
```
num_str = num_str + ch
```
| 要素 | 制約 |
|------|------|
| **左辺** | `num_str`LoopState carrier |
| **右辺** | `BinaryOp(Add, num_str, ch)` のみ |
| **ch** | body-local または captured 変数 |
### 2.2 許可パターンの形式定義
```rust
// 許可: num_str = num_str + ch
UpdatePattern::StringAppend {
carrier: "num_str",
append_source: Variable("ch"),
}
// 内部表現
BinaryOp {
op: Add,
lhs: Variable { name: carrier_name }, // 同じキャリア
rhs: Variable { name: append_var }, // body-local/captured
}
```
### 2.3 非対象パターン(将来フェーズ候補)
| パターン | 理由 | 候補フェーズ |
|----------|------|-------------|
| `ch + num_str` | 逆順 concat | Phase 246 |
| `num_str + num_str` | 自己 concat | Phase 247 |
| `a + b + c` | 三項以上 | Phase 247 |
| `num_str.append(ch)` | MethodCall | Phase 248 |
### 2.4 CarrierUpdateEmitter への統合案
**Option A**: 既存 UpdateExpr に `StringConcat` variant 追加
```rust
pub enum UpdateExpr {
// 既存
Increment { carrier: String, amount: i64 },
Accumulate { carrier: String, source: ValueId },
// 新規追加 (245B)
StringAppend { carrier: String, append_source: String },
}
```
**Option B**: Generic `BinaryUpdate` で統一
```rust
pub enum UpdateExpr {
BinaryUpdate {
carrier: String,
op: BinaryOperator,
rhs: UpdateRhs,
},
}
pub enum UpdateRhs {
Variable(String),
Constant(ConstValue),
Carrier(String),
}
```
**推奨**: Option A最小変更、明示的
---
## 3. num_str キャリアの Contract / Invariant
### 3.1 Loop Entry Contract
```
PRE:
- num_str = "" (空文字で初期化済み)
- p = 開始位置 (valid index)
- s = 対象文字列 (immutable)
```
### 3.2 Loop Iteration Invariant
```
INV:
- num_str = s[start_p..p] の digit 部分文字列
- p は monotonic increasing (p' > p)
- num_str.length() == p - start_p
```
### 3.3 Loop Exit Contract
```
POST (break):
- num_str = parse された digit 列
- p = 最初の non-digit 位置
- num_str == s[start_p..p]
POST (natural exit):
- num_str = s[start_p..len] 全て digit
- p == len
```
### 3.4 ExitMeta 構造
```rust
ExitMeta {
exit_values: vec![
("p".to_string(), p_final), // loop counter
("num_str".to_string(), num_str_final), // string carrier
],
}
```
---
## 4. テストケース
### 4.1 E2E テスト(正常系)
| 入力 | 期待 num_str | 期待 RC |
|------|-------------|---------|
| `"0"` | `"0"` | 0 |
| `"42"` | `"42"` | 0 |
| `"123456"` | `"123456"` | 0 |
| `"007"` | `"007"` | 0 (先頭 0 許容) |
### 4.2 E2E テスト(部分マッチ系)
| 入力 | 期待 num_str | 期待 p | 備考 |
|------|-------------|--------|------|
| `"7z"` | `"7"` | 1 | 1文字で break |
| `"123abc"` | `"123"` | 3 | 3文字で break |
| `"abc"` | `""` | 0 | 即 break |
### 4.3 JoinIR 構造テスト
```rust
#[test]
fn test_parse_number_string_carrier() {
// Given: _parse_number loop
// When: JoinIR generated
// Then:
// - num_str is in CarrierInfo with role=LoopState
// - UpdateExpr::StringAppend for num_str exists
// - ExitMeta contains num_str exit value
}
```
### 4.4 UpdateExpr 検証テスト
```rust
#[test]
fn test_string_append_update_expr() {
// Given: num_str = num_str + ch
// When: UpdateExpr extracted
// Then:
// - UpdateExpr::StringAppend { carrier: "num_str", append_source: "ch" }
// - Exactly 1 StringAppend for num_str
}
```
---
## 5. 制約と非目標
### 5.1 Phase 245B の制約
| 制約 | 理由 |
|------|------|
| `_parse_number` のみ対象 | スコープ限定、段階的実装 |
| 新パターン追加なし | P2 + 既存インフラ活用 |
| by-name hardcode 禁止 | CarrierInfo/UpdateExpr で区別 |
| StringAppend 1形式のみ | 最小セットから開始 |
### 5.2 非目標245B では実装しない)
- `_atoi` / `_atof_loop` との共通化
- Pattern 3/4 への文字列キャリア拡張
- MethodCall 形式の append (`num_str.append(ch)`)
- 逆順 concat (`ch + num_str`)
- 三項以上の concat
### 5.3 後続フェーズ候補
| フェーズ | 内容 |
|---------|------|
| **246** | `_atoi` / `_atof_loop` との整合性 |
| **247** | Pattern 3/4 に文字列キャリア拡張 |
| **248** | MethodCall append 対応 |
| **249** | 逆順・三項 concat 対応 |
---
## 6. 実装ロードマップ
### 6.1 Phase 245B-1: UpdateExpr 拡張
1. `UpdateExpr::StringAppend` variant 追加
2. CarrierUpdateEmitter に StringAppend 検出ロジック
3. ユニットテスト 3-5 件
### 6.2 Phase 245B-2: CarrierInfo 統合
1. String 型キャリアの role=LoopState 対応
2. ExitMeta に string carrier 含める
3. Header/Exit PHI に string value 通す
### 6.3 Phase 245B-3: Pattern 2 統合
1. `loop_with_break_minimal.rs` で StringAppend 処理
2. E2E テスト実装
3. `_parse_number` テストケース通す
### 6.4 完了条件
- [ ] `cargo build --release` 成功
- [ ] 全テスト PASS911+
- [ ] `_parse_number` E2E 4+ ケース PASS
- [ ] UpdateExpr::StringAppend ユニットテスト PASS
- [ ] by-name hardcode なし
---
## 7. リスク評価
| リスク | 影響度 | 軽減策 |
|--------|--------|--------|
| String PHI の型不整合 | 中 | 既存 String carrier テスト確認 |
| UpdateExpr 検出失敗 | 中 | 段階的検出ロジック |
| Pattern 2 退行 | 低 | 911 テスト PASS 維持 |
| 将来の拡張困難 | 低 | Option A 設計で拡張可能 |
---
## 8. 承認チェックリスト
- [ ] num_str の役割Section 1確認
- [ ] UpdateExpr パターンSection 2確認
- [ ] Contract/InvariantSection 3確認
- [ ] テストケースSection 4確認
- [ ] 制約と非目標Section 5確認
- [ ] 実装ロードマップSection 6確認
**承認後、Phase 245B-1 実装開始!**
---
*Document created: Phase 245B Design*
*Author: Claude Code Session*
*Date: 2025-12-11*