gcf-python 0.1.3__py3-none-any.whl → 0.3.0__py3-none-any.whl

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.
gcf/decode.py CHANGED
@@ -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":
gcf/encode.py CHANGED
@@ -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"
gcf/generic.py CHANGED
@@ -59,6 +59,10 @@ def _encode_array(items: list, name: str, lines: list[str], depth: int) -> None:
59
59
 
60
60
  if _is_uniform_dict_list(items):
61
61
  _encode_tabular(items, name, lines, depth)
62
+ elif all(not isinstance(item, (dict, list)) for item in items):
63
+ # Primitive array: inline as comma-separated values.
64
+ vals = ",".join(_format_value(item) for item in items)
65
+ lines.append(f"{prefix}{name}[{len(items)}]: {vals}")
62
66
  else:
63
67
  lines.append(f"{prefix}## {name} [{len(items)}]")
64
68
  for i, item in enumerate(items):
@@ -142,7 +146,7 @@ def _format_value(value: Any) -> str:
142
146
  return str(value)
143
147
  s = str(value)
144
148
  if "|" in s or "\n" in s or s == "":
145
- escaped = s.replace("\\", "\\\\").replace('"', '\\"')
149
+ escaped = s.replace("\\", "\\\\").replace('"', '\\"').replace("\n", "\\n")
146
150
  return f'"{escaped}"'
147
151
  return s
148
152
 
gcf/session.py CHANGED
@@ -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)
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: gcf-python
3
- Version: 0.1.3
3
+ Version: 0.3.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
 
@@ -0,0 +1,14 @@
1
+ gcf/__init__.py,sha256=epW-CoQh3RBrnF3-lWSiPS-DI4HUlRRe8q9JeLPRxwE,1541
2
+ gcf/cli.py,sha256=2hSTBqiYcn1_EgGXuO65MHiEGh0C4DRMvspTd2zUaso,4258
3
+ gcf/constants.py,sha256=cmZ8YJSOB0im_eyfN8v4UvrLpBC6Fuf4cfcKZGbutxY,638
4
+ gcf/decode.py,sha256=48G6XmBilGYl0-c_Xy7MX0iTwOxGW9bn3wC4UyNApc4,5465
5
+ gcf/delta.py,sha256=xU0ujtSq1iF7yU8yk_WNQKh8iove-WUV_nKSuvW1XVk,1656
6
+ gcf/encode.py,sha256=Oljb1r5b7SHmng1XYvcvuJMpaRfJVJ81VOyocPf_kAs,2915
7
+ gcf/generic.py,sha256=Nf1Ii0pYS0dgZ4o2ghl-3Qhnms-kCKY2D3q-3-oGpe8,5454
8
+ gcf/session.py,sha256=4_ARRL06Tg2CI8D2eyi0V5nFphFAFMfOKBXYnAbI6Nk,4690
9
+ gcf/types.py,sha256=AWm-LQoSqLHAYtEjcAxWQZqJ4JXqNreLUKO2mJFgNMA,1465
10
+ gcf_python-0.3.0.dist-info/METADATA,sha256=-EBHKi9DP35REiBPf40__SJ_8QPxk41uRPCiCil-ZRM,7929
11
+ gcf_python-0.3.0.dist-info/WHEEL,sha256=mffPy8wBnZQn2VnJUU5jE99KsxaSfiyMHV9Yt0aLVxs,87
12
+ gcf_python-0.3.0.dist-info/entry_points.txt,sha256=aFT6gqlkh8iGfM8cblE-LUMxHH08_v71IIoZtDdRIVA,37
13
+ gcf_python-0.3.0.dist-info/licenses/LICENSE,sha256=txSvg3E4LugiB7MOOTci6WKd6wMOrOJTvaITeFJ2SgU,1074
14
+ gcf_python-0.3.0.dist-info/RECORD,,
@@ -1,14 +0,0 @@
1
- gcf/__init__.py,sha256=epW-CoQh3RBrnF3-lWSiPS-DI4HUlRRe8q9JeLPRxwE,1541
2
- gcf/cli.py,sha256=2hSTBqiYcn1_EgGXuO65MHiEGh0C4DRMvspTd2zUaso,4258
3
- gcf/constants.py,sha256=cmZ8YJSOB0im_eyfN8v4UvrLpBC6Fuf4cfcKZGbutxY,638
4
- gcf/decode.py,sha256=gkBW9fmcurQ9bfcDXbaCOWemMmljo_MiZMj3-1rYvsw,5198
5
- gcf/delta.py,sha256=xU0ujtSq1iF7yU8yk_WNQKh8iove-WUV_nKSuvW1XVk,1656
6
- gcf/encode.py,sha256=WZCRv1Vj2PDTYvBgv__2P8GHh8FapxfAZmt5zo0k9Jc,2700
7
- gcf/generic.py,sha256=xOaACtsQSgp5kT6N8Lx47Z_YtqRHnSKsMf5Fs-gHcvY,5179
8
- gcf/session.py,sha256=F8OTJCMRMCY7Yzcvd7aU8wlbER0LZAMUcB4W9kHeSq4,4508
9
- gcf/types.py,sha256=AWm-LQoSqLHAYtEjcAxWQZqJ4JXqNreLUKO2mJFgNMA,1465
10
- gcf_python-0.1.3.dist-info/METADATA,sha256=Y9vxOKcBtYAP9AGnBw-qA2IXpQWekmWNS9eBOV_7w6c,7917
11
- gcf_python-0.1.3.dist-info/WHEEL,sha256=mffPy8wBnZQn2VnJUU5jE99KsxaSfiyMHV9Yt0aLVxs,87
12
- gcf_python-0.1.3.dist-info/entry_points.txt,sha256=aFT6gqlkh8iGfM8cblE-LUMxHH08_v71IIoZtDdRIVA,37
13
- gcf_python-0.1.3.dist-info/licenses/LICENSE,sha256=txSvg3E4LugiB7MOOTci6WKd6wMOrOJTvaITeFJ2SgU,1074
14
- gcf_python-0.1.3.dist-info/RECORD,,