aether-code 0.16.3 → 0.17.0

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.
@@ -26,7 +26,7 @@ import {
26
26
  import readline from "node:readline";
27
27
  import { c, errorLine, divider } from "../src/render.js";
28
28
 
29
- const VERSION = "0.16.3";
29
+ const VERSION = "0.17.0";
30
30
 
31
31
  /**
32
32
  * Try to start MCP servers from ~/.aether/mcp.json. Returns a started
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "aether-code",
3
- "version": "0.16.3",
3
+ "version": "0.17.0",
4
4
  "description": "Uncensored AI coding agent for your terminal — Claude Code alternative with MCP support. Reads code, writes files, runs commands. Drives IDA Pro, Roblox Studio, Wireshark, Blender, and any MCP server. No refusal layer.",
5
5
  "homepage": "https://trynoguard.com",
6
6
  "repository": {
package/src/agent.js CHANGED
@@ -6,7 +6,7 @@ import { agentTurnStream, AetherError } from "./api.js";
6
6
  import { TOOL_DEFINITIONS, executeTool } from "./tools.js";
7
7
  import { unnamespaceToolName } from "./mcp.js";
8
8
  import { loadAllSkills, selectSkills, renderSkillsBlock } from "./skills.js";
9
- import { c, divider, turn, toolLabel, toolSummary, stripModelTokens, errorLine } from "./render.js";
9
+ import { c, divider, turn, toolLabel, toolSummary, makeTokenStripper, errorLine } from "./render.js";
10
10
 
11
11
  const DEFAULT_MAX_TURNS = 25;
12
12
 
@@ -59,6 +59,7 @@ export async function runAgent({
59
59
  // calling a particular tool (i.e. the `name` arrives in the stream).
60
60
  const announced = new Set();
61
61
  let lastWasText = false;
62
+ const stripper = makeTokenStripper();
62
63
 
63
64
  // Select skills for this turn against the current user prompt + any
64
65
  // paths the model has read so far. Prepend the matching skills' bodies
@@ -73,8 +74,9 @@ export async function runAgent({
73
74
  messages: turnMessages,
74
75
  tools,
75
76
  onDelta: (text) => {
76
- // Strip any leaked model channel/control tokens before display.
77
- const clean = stripModelTokens(text);
77
+ // Buffered strip of leaked model channel/control tokens (which can
78
+ // be split across stream chunks) before display.
79
+ const clean = stripper.push(text);
78
80
  if (!clean) return;
79
81
  if (!lastWasText) {
80
82
  process.stdout.write(" ");
@@ -100,7 +102,12 @@ export async function runAgent({
100
102
  throw err;
101
103
  }
102
104
 
103
- // End-of-turn newline + cost meter
105
+ // Flush any held-back partial token, then close the line.
106
+ const tail = stripper.flush();
107
+ if (tail) {
108
+ if (!lastWasText) { process.stdout.write(" "); lastWasText = true; }
109
+ process.stdout.write(tail);
110
+ }
104
111
  if (lastWasText) process.stdout.write("\n");
105
112
  totalCredits += res.creditsCharged ?? 0;
106
113
  totalIn += res.usage?.prompt_tokens ?? 0;
package/src/render.js CHANGED
@@ -92,11 +92,40 @@ export function toolSummary(name, result) {
92
92
  // Strip model "harmony"/channel control tokens (<|channel|>, <|message|>,
93
93
  // <|tool_response|>, <channel|>, …) that occasionally leak into the text
94
94
  // stream. Belt-and-suspenders alongside the server-side scrub.
95
+ // Only strips tokens containing a PIPE — the harmony/channel control tokens
96
+ // (<|channel|>, <|tool_response|>, <channel|>) always have one. Real code like
97
+ // <div>, Vec<T>, a < b has no pipe and is left untouched.
98
+ const MODEL_TOKEN_RE = /<\|[a-z_]*\|?>|<[a-z_]+\|>/gi;
99
+
95
100
  export function stripModelTokens(text) {
96
- return text
97
- .replace(/<\|[a-z_]*\|?>/gi, "")
98
- .replace(/<\/?[a-z_]*\|>/gi, "")
99
- .replace(/<\|[a-z_]*>/gi, "");
101
+ return text.replace(MODEL_TOKEN_RE, "");
102
+ }
103
+
104
+ // Streaming-safe stripper: the leaked tokens (<|channel|>, <|tool_response|>, …)
105
+ // can be split across stream chunks ("<chann" then "el|>"), which a per-delta
106
+ // regex misses. This buffers any trailing "<…" that might be the start of a
107
+ // token and only emits it once it's confirmed not-a-token (or on flush).
108
+ export function makeTokenStripper() {
109
+ let buf = "";
110
+ return {
111
+ push(text) {
112
+ buf = (buf + text).replace(MODEL_TOKEN_RE, "");
113
+ const partial = buf.match(/<[|a-z_/]*$/i); // possible token start at the tail
114
+ if (partial) {
115
+ const emit = buf.slice(0, partial.index);
116
+ buf = buf.slice(partial.index);
117
+ return emit;
118
+ }
119
+ const emit = buf;
120
+ buf = "";
121
+ return emit;
122
+ },
123
+ flush() {
124
+ const out = buf.replace(MODEL_TOKEN_RE, "");
125
+ buf = "";
126
+ return out;
127
+ },
128
+ };
100
129
  }
101
130
 
102
131
  export function toolResult(text, ok = true) {
package/src/repl.js CHANGED
@@ -17,7 +17,7 @@ import { c, errorLine } from "./render.js";
17
17
  import { checkForUpdate } from "./update-check.js";
18
18
  import { promptBoxed, EXIT_SIGNAL } from "./ink-input.js";
19
19
 
20
- const VERSION = "0.16.3";
20
+ const VERSION = "0.17.0";
21
21
  const MODEL_NAME = "Aether Core";
22
22
 
23
23
  const SHORTCUTS = `
@@ -100,28 +100,32 @@ export async function runRepl({ cwd: initialCwd, autoYes: initialAutoYes, maxTur
100
100
  process.env.AETHER_NO_INK !== "1" &&
101
101
  (process.env.AETHER_INK === "1" || !legacyWinConsole);
102
102
  let inkBroken = false;
103
- let rl = null;
104
103
 
105
104
  // The Ink box carries its own status bar; the plain prompt doesn't, so show a
106
105
  // one-line hint up front when we won't be using Ink.
107
106
  if (!useInk) console.log(` ${c.cyan("/help")}${c.dim(" shortcuts")} ${c.cyan("/exit")}${c.dim(" quit")}\n`);
108
107
 
109
- function ensureReadline() {
110
- if (rl) return rl;
111
- rl = readline.createInterface({ input: process.stdin, output: process.stdout, historySize: 200 });
112
- let lastSigint = 0;
113
- rl.on("SIGINT", () => {
114
- const now = Date.now();
115
- if (now - lastSigint < 1500) { console.log(c.gray("\nbye.")); rl.close(); process.exit(0); }
116
- lastSigint = now;
117
- console.log(c.gray(`\n(Press Ctrl+C again within 1.5s to exit, or type ${c.cyan("/exit")})`));
118
- });
119
- return rl;
120
- }
121
-
108
+ // Fresh readline interface PER PROMPT. A single persistent interface
109
+ // conflicted with the per-confirmation readline interfaces the tools open
110
+ // during a turn (each [y/N] prompt). When a tool's interface closed it left
111
+ // the REPL's stdin dead — so after finishing a request aether printed the
112
+ // prompt and immediately EXITED instead of waiting for the next message.
113
+ // Opening a fresh one each prompt guarantees only one interface exists at a
114
+ // time, so the REPL keeps running until the user /exits. Returns EXIT_SIGNAL
115
+ // on a double Ctrl+C.
122
116
  function readlineQuestion() {
123
- const r = ensureReadline();
124
- return new Promise((resolve) => r.question(c.magenta("> "), (ans) => resolve(ans)));
117
+ return new Promise((resolve) => {
118
+ const r = readline.createInterface({ input: process.stdin, output: process.stdout, historySize: 200 });
119
+ let lastSigint = 0;
120
+ r.on("SIGINT", () => {
121
+ const now = Date.now();
122
+ if (now - lastSigint < 1500) { r.close(); resolve(EXIT_SIGNAL); return; }
123
+ lastSigint = now;
124
+ process.stdout.write(c.gray(`\n(Press Ctrl+C again within 1.5s to exit, or type ${c.cyan("/exit")})\n`));
125
+ r.prompt();
126
+ });
127
+ r.question(c.magenta("> "), (ans) => { r.close(); resolve(ans); });
128
+ });
125
129
  }
126
130
 
127
131
  // Returns the next raw input line, or EXIT_SIGNAL to quit.
@@ -143,7 +147,7 @@ export async function runRepl({ cwd: initialCwd, autoYes: initialAutoYes, maxTur
143
147
 
144
148
  while (true) {
145
149
  const raw = await nextLine();
146
- if (raw === EXIT_SIGNAL) { console.log(c.gray("bye.")); if (rl) rl.close(); return; }
150
+ if (raw === EXIT_SIGNAL) { console.log(c.gray("bye.")); return; }
147
151
  const line = (raw ?? "").trim();
148
152
  if (!line) continue;
149
153
 
@@ -155,7 +159,7 @@ export async function runRepl({ cwd: initialCwd, autoYes: initialAutoYes, maxTur
155
159
  // Slash command?
156
160
  if (line.startsWith("/") || line === "?") {
157
161
  const handled = await handleSlash(line, state);
158
- if (handled === "exit") { if (rl) rl.close(); return; }
162
+ if (handled === "exit") return;
159
163
  printStatusLine(state);
160
164
  continue;
161
165
  }