gcf-python 0.1.3__tar.gz → 0.2.0__tar.gz

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.
Files changed (24) hide show
  1. {gcf_python-0.1.3 → gcf_python-0.2.0}/CHANGELOG.md +13 -0
  2. {gcf_python-0.1.3 → gcf_python-0.2.0}/PKG-INFO +3 -3
  3. {gcf_python-0.1.3 → gcf_python-0.2.0}/README.md +2 -2
  4. {gcf_python-0.1.3 → gcf_python-0.2.0}/pyproject.toml +1 -1
  5. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/decode.py +7 -0
  6. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/encode.py +13 -7
  7. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/generic.py +1 -1
  8. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/session.py +13 -7
  9. {gcf_python-0.1.3 → gcf_python-0.2.0}/tests/test_decode.py +4 -4
  10. {gcf_python-0.1.3 → gcf_python-0.2.0}/tests/test_encode.py +6 -5
  11. {gcf_python-0.1.3 → gcf_python-0.2.0}/.github/workflows/ci.yml +0 -0
  12. {gcf_python-0.1.3 → gcf_python-0.2.0}/.github/workflows/publish.yml +0 -0
  13. {gcf_python-0.1.3 → gcf_python-0.2.0}/.gitignore +0 -0
  14. {gcf_python-0.1.3 → gcf_python-0.2.0}/LICENSE +0 -0
  15. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/__init__.py +0 -0
  16. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/cli.py +0 -0
  17. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/constants.py +0 -0
  18. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/delta.py +0 -0
  19. {gcf_python-0.1.3 → gcf_python-0.2.0}/src/gcf/types.py +0 -0
  20. {gcf_python-0.1.3 → gcf_python-0.2.0}/tests/__init__.py +0 -0
  21. {gcf_python-0.1.3 → gcf_python-0.2.0}/tests/test_delta.py +0 -0
  22. {gcf_python-0.1.3 → gcf_python-0.2.0}/tests/test_generic.py +0 -0
  23. {gcf_python-0.1.3 → gcf_python-0.2.0}/tests/test_roundtrip.py +0 -0
  24. {gcf_python-0.1.3 → gcf_python-0.2.0}/tests/test_session.py +0 -0
@@ -1,5 +1,18 @@
1
1
  # Changelog
2
2
 
3
+ ## v0.2.0 (2026-06-05)
4
+
5
+ - **Breaking**: `encode()` now emits `edges=N` in header line
6
+ - **Breaking**: `encode()` now emits `## edges [N]` section header (was `## edges`)
7
+ - `decode()` updated to parse `## edges [N]` format (strips bracket suffix)
8
+ - Session encoder updated to emit new edge count format
9
+
10
+ ## v0.1.3 (2026-06-04)
11
+
12
+ - Docs: update README for PyPI discoverability (gcformat.com, proxy, vs-toon links)
13
+ - Fix: decoder rejects headers missing required `tool` field (conformance)
14
+ - Fix: escape newlines as `\n` in quoted strings in `encode_generic`
15
+
3
16
  ## v0.1.2 (2026-06-04)
4
17
 
5
18
  - Fix: escape `"` inside quoted strings in `encode_generic`
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: gcf-python
3
- Version: 0.1.3
3
+ Version: 0.2.0
4
4
  Summary: Python implementation of GCF (Graph Compact Format): token-optimized wire format for LLM tool responses
5
5
  Project-URL: Homepage, https://github.com/blackwell-systems/gcf-python
6
6
  Project-URL: Documentation, https://blackwell-systems.github.io/gcf/
@@ -86,12 +86,12 @@ output = encode(p)
86
86
 
87
87
  Output:
88
88
  ```
89
- GCF tool=context_for_task budget=5000 tokens=1847 symbols=2
89
+ GCF tool=context_for_task budget=5000 tokens=1847 symbols=2 edges=1
90
90
  ## targets
91
91
  @0 fn pkg.AuthMiddleware 0.78 lsp_resolved
92
92
  ## related
93
93
  @1 fn pkg.NewServer 0.54 lsp_resolved
94
- ## edges
94
+ ## edges [1]
95
95
  @0<@1 calls
96
96
  ```
97
97
 
@@ -61,12 +61,12 @@ output = encode(p)
61
61
 
62
62
  Output:
63
63
  ```
64
- GCF tool=context_for_task budget=5000 tokens=1847 symbols=2
64
+ GCF tool=context_for_task budget=5000 tokens=1847 symbols=2 edges=1
65
65
  ## targets
66
66
  @0 fn pkg.AuthMiddleware 0.78 lsp_resolved
67
67
  ## related
68
68
  @1 fn pkg.NewServer 0.54 lsp_resolved
69
- ## edges
69
+ ## edges [1]
70
70
  @0<@1 calls
71
71
  ```
72
72
 
@@ -4,7 +4,7 @@ build-backend = "hatchling.build"
4
4
 
5
5
  [project]
6
6
  name = "gcf-python"
7
- version = "0.1.3"
7
+ version = "0.2.0"
8
8
  description = "Python implementation of GCF (Graph Compact Format): token-optimized wire format for LLM tool responses"
9
9
  readme = "README.md"
10
10
  license = {text = "MIT"}
@@ -34,6 +34,9 @@ def decode(input_text: str) -> Payload:
34
34
  raise DecodeError(f"invalid header, expected 'GCF ...' got {header!r}")
35
35
  _parse_header(header[4:], p)
36
36
 
37
+ if not p.tool:
38
+ raise DecodeError("header missing required 'tool' field")
39
+
37
40
  # Parse body: symbols and edges.
38
41
  symbols: list[Symbol] = []
39
42
  sym_by_id: dict[int, Symbol] = {}
@@ -48,6 +51,10 @@ def decode(input_text: str) -> Payload:
48
51
  # Group header.
49
52
  if line.startswith("## "):
50
53
  group = line[3:]
54
+ # Strip bracket suffix: "edges [200]" -> "edges"
55
+ bracket_idx = group.find(" [")
56
+ if bracket_idx >= 0:
57
+ group = group[:bracket_idx]
51
58
  in_edges = group == "edges"
52
59
  if not in_edges:
53
60
  if group == "targets":
@@ -17,17 +17,23 @@ def encode(p: Payload) -> str:
17
17
  """
18
18
  parts: list[str] = []
19
19
 
20
- # Header line.
21
- header = f"GCF tool={p.tool} budget={p.token_budget} tokens={p.tokens_used} symbols={len(p.symbols)}"
22
- if p.pack_root:
23
- header += f" pack_root={p.pack_root}"
24
- parts.append(header)
25
-
26
20
  # Build symbol index for edge references.
27
21
  sym_index: dict[str, int] = {}
28
22
  for i, s in enumerate(p.symbols):
29
23
  sym_index[s.qualified_name] = i
30
24
 
25
+ # Count valid edges (both endpoints in symbol index).
26
+ valid_edges = sum(
27
+ 1 for e in p.edges
28
+ if e.source in sym_index and e.target in sym_index
29
+ )
30
+
31
+ # Header line.
32
+ header = f"GCF tool={p.tool} budget={p.token_budget} tokens={p.tokens_used} symbols={len(p.symbols)} edges={valid_edges}"
33
+ if p.pack_root:
34
+ header += f" pack_root={p.pack_root}"
35
+ parts.append(header)
36
+
31
37
  # Group symbols by distance.
32
38
  groups = _group_by_distance(p.symbols)
33
39
  group_names = ["targets", "related", "extended"]
@@ -58,7 +64,7 @@ def encode(p: Payload) -> str:
58
64
  if e.status and e.status != "unchanged":
59
65
  line += f" {e.status}"
60
66
  edge_lines.append(line)
61
- parts.append("## edges")
67
+ parts.append(f"## edges [{len(edge_lines)}]")
62
68
  parts.extend(edge_lines)
63
69
 
64
70
  return "\n".join(parts) + "\n"
@@ -142,7 +142,7 @@ def _format_value(value: Any) -> str:
142
142
  return str(value)
143
143
  s = str(value)
144
144
  if "|" in s or "\n" in s or s == "":
145
- escaped = s.replace("\\", "\\\\").replace('"', '\\"')
145
+ escaped = s.replace("\\", "\\\\").replace('"', '\\"').replace("\n", "\\n")
146
146
  return f'"{escaped}"'
147
147
  return s
148
148
 
@@ -77,20 +77,26 @@ def encode_with_session(p: Payload, sess: Session | None = None) -> str:
77
77
 
78
78
  parts: list[str] = []
79
79
 
80
+ # Build local ID mapping for this response.
81
+ local_index: dict[str, int] = {}
82
+ for i, s in enumerate(p.symbols):
83
+ local_index[s.qualified_name] = i
84
+
85
+ # Count valid edges.
86
+ valid_edges = sum(
87
+ 1 for e in p.edges
88
+ if e.source in local_index and e.target in local_index
89
+ )
90
+
80
91
  # Header with session=true marker.
81
92
  header = (
82
93
  f"GCF tool={p.tool} budget={p.token_budget} tokens={p.tokens_used} "
83
- f"symbols={len(p.symbols)} session=true"
94
+ f"symbols={len(p.symbols)} edges={valid_edges} session=true"
84
95
  )
85
96
  if p.pack_root:
86
97
  header += f" pack_root={p.pack_root}"
87
98
  parts.append(header)
88
99
 
89
- # Build local ID mapping for this response.
90
- local_index: dict[str, int] = {}
91
- for i, s in enumerate(p.symbols):
92
- local_index[s.qualified_name] = i
93
-
94
100
  # Track which symbols are new (need full declaration).
95
101
  new_symbols: list[Symbol] = []
96
102
 
@@ -122,7 +128,7 @@ def encode_with_session(p: Payload, sess: Session | None = None) -> str:
122
128
 
123
129
  # Edges section.
124
130
  if p.edges:
125
- parts.append("## edges")
131
+ parts.append(f"## edges [{valid_edges}]")
126
132
  for e in p.edges:
127
133
  src_idx = local_index.get(e.source)
128
134
  tgt_idx = local_index.get(e.target)
@@ -13,7 +13,7 @@ def test_decode_basic_payload():
13
13
  "@0 fn pkg.AuthMiddleware 0.78 lsp_resolved\n"
14
14
  "## related\n"
15
15
  "@1 fn pkg.NewServer 0.54 lsp_resolved\n"
16
- "## edges\n"
16
+ "## edges [1]\n"
17
17
  "@0<@1 calls\n"
18
18
  )
19
19
 
@@ -93,7 +93,7 @@ def test_decode_edge_with_status():
93
93
  "## targets\n"
94
94
  "@0 fn a.A 0.90 x\n"
95
95
  "@1 fn b.B 0.80 x\n"
96
- "## edges\n"
96
+ "## edges [1]\n"
97
97
  "@0<@1 calls added\n"
98
98
  )
99
99
  p = decode(input_text)
@@ -171,7 +171,7 @@ def test_decode_edge_missing_separator():
171
171
  "## targets\n"
172
172
  "@0 fn a.A 0.90 x\n"
173
173
  "@1 fn b.B 0.80 x\n"
174
- "## edges\n"
174
+ "## edges [1]\n"
175
175
  "@0@1 calls\n"
176
176
  )
177
177
  with pytest.raises(DecodeError, match="missing '<' separator"):
@@ -184,7 +184,7 @@ def test_decode_edge_unknown_symbol():
184
184
  "GCF tool=test budget=100 tokens=50 symbols=1\n"
185
185
  "## targets\n"
186
186
  "@0 fn a.A 0.90 x\n"
187
- "## edges\n"
187
+ "## edges [1]\n"
188
188
  "@0<@99 calls\n"
189
189
  )
190
190
  with pytest.raises(DecodeError, match="unknown symbol id"):
@@ -32,12 +32,12 @@ def test_encode_basic_payload():
32
32
 
33
33
  output = encode(p)
34
34
  expected = (
35
- "GCF tool=context_for_task budget=5000 tokens=1847 symbols=2\n"
35
+ "GCF tool=context_for_task budget=5000 tokens=1847 symbols=2 edges=1\n"
36
36
  "## targets\n"
37
37
  "@0 fn pkg.AuthMiddleware 0.78 lsp_resolved\n"
38
38
  "## related\n"
39
39
  "@1 fn pkg.NewServer 0.54 lsp_resolved\n"
40
- "## edges\n"
40
+ "## edges [1]\n"
41
41
  "@0<@1 calls\n"
42
42
  )
43
43
  assert output == expected
@@ -175,8 +175,9 @@ def test_encode_skips_edges_with_missing_symbols():
175
175
  )
176
176
  output = encode(p)
177
177
  # Section header is emitted (matches Go), but no edge lines beneath it
178
- assert "## edges" in output
179
- lines_after_edges = output.split("## edges\n")[1]
178
+ assert "## edges [0]" in output
179
+ assert "edges=0" in output
180
+ lines_after_edges = output.split("## edges [0]\n")[1]
180
181
  assert lines_after_edges.strip() == ""
181
182
 
182
183
 
@@ -184,4 +185,4 @@ def test_encode_empty_payload():
184
185
  """Empty payload produces only header."""
185
186
  p = Payload(tool="test", token_budget=100, tokens_used=0)
186
187
  output = encode(p)
187
- assert output == "GCF tool=test budget=100 tokens=0 symbols=0\n"
188
+ assert output == "GCF tool=test budget=100 tokens=0 symbols=0 edges=0\n"
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes