getaimeter 0.7.2 → 0.7.4

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 (3) hide show
  1. package/package.json +1 -1
  2. package/service.js +29 -22
  3. package/watcher.js +35 -15
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "getaimeter",
3
- "version": "0.7.2",
3
+ "version": "0.7.4",
4
4
  "description": "Track AI coding costs across Claude, Cursor, Codex, and Gemini. Optimization recommendations that cut costs by 30%.",
5
5
  "bin": {
6
6
  "aimeter": "cli.js"
package/service.js CHANGED
@@ -98,37 +98,44 @@ WshShell.Run """${nodePath}"" ""${script}"" watch", 0, False
98
98
  * These are proper Windows shortcuts with icon and description.
99
99
  */
100
100
  function createWindowsShortcuts(nodePath, script) {
101
- const iconPath = path.join(__dirname, 'icon.ico').replace(/\\/g, '\\\\');
101
+ const { execSync } = require('child_process');
102
+ const iconPath = path.join(__dirname, 'icon.ico');
103
+
102
104
  const locations = [
103
- { name: 'Desktop', dir: path.join(os.homedir(), 'Desktop') },
104
- { name: 'Start Menu', dir: path.join(os.homedir(), 'AppData', 'Roaming', 'Microsoft', 'Windows', 'Start Menu', 'Programs') },
105
+ path.join(os.homedir(), 'Desktop', 'AIMeter.lnk'),
106
+ path.join(os.homedir(), 'AppData', 'Roaming', 'Microsoft', 'Windows', 'Start Menu', 'Programs', 'AIMeter.lnk'),
105
107
  ];
106
108
 
107
- for (const loc of locations) {
108
- const lnkPath = path.join(loc.dir, 'AIMeter.lnk');
109
- // PowerShell script to create a .lnk shortcut
110
- const ps = `
111
- $ws = New-Object -ComObject WScript.Shell
112
- $s = $ws.CreateShortcut('${lnkPath.replace(/'/g, "''")}')
113
- $s.TargetPath = '${nodePath.replace(/\\\\/g, '\\').replace(/'/g, "''")}'
114
- $s.Arguments = '"${script.replace(/\\\\/g, '\\').replace(/'/g, "''")}" start'
115
- $s.WorkingDirectory = '${os.homedir().replace(/'/g, "''")}'
116
- $s.Description = 'AIMeter - AI coding cost optimizer'
117
- $s.WindowStyle = 7
118
- $iconFile = '${iconPath.replace(/\\\\/g, '\\').replace(/'/g, "''")}'
119
- if (Test-Path $iconFile) { $s.IconLocation = $iconFile }
120
- $s.Save()
121
- `.trim();
109
+ // Write a temporary PS1 file to avoid all escaping issues
110
+ const ps1Path = path.join(os.tmpdir(), 'aimeter-shortcut.ps1');
111
+
112
+ for (const lnkPath of locations) {
113
+ // Use single quotes in PS1 with doubled single-quote escaping
114
+ const ps1Content = [
115
+ '$ws = New-Object -ComObject WScript.Shell',
116
+ `$s = $ws.CreateShortcut('${lnkPath.replace(/'/g, "''")}')`,
117
+ `$s.TargetPath = '${nodePath.replace(/\\\\/g, '\\').replace(/'/g, "''")}'`,
118
+ `$s.Arguments = [char]34 + '${script.replace(/\\\\/g, '\\').replace(/'/g, "''")}' + [char]34 + ' start'`,
119
+ `$s.WorkingDirectory = '${os.homedir().replace(/'/g, "''")}'`,
120
+ `$s.Description = 'AIMeter - AI coding cost optimizer'`,
121
+ `$s.WindowStyle = 7`,
122
+ `$icon = '${iconPath.replace(/'/g, "''")}'`,
123
+ `if (Test-Path $icon) { $s.IconLocation = $icon }`,
124
+ `$s.Save()`,
125
+ ].join('\n');
122
126
 
123
127
  try {
124
- require('child_process').execSync(
125
- `powershell -NoProfile -ExecutionPolicy Bypass -Command "${ps.replace(/"/g, '\\"')}"`,
126
- { stdio: 'ignore', timeout: 10000 }
127
- );
128
+ fs.writeFileSync(ps1Path, ps1Content, 'utf8');
129
+ execSync(`powershell -NoProfile -ExecutionPolicy Bypass -File "${ps1Path}"`, {
130
+ stdio: 'ignore',
131
+ timeout: 10000,
132
+ });
128
133
  } catch {
129
134
  // Non-critical — shortcuts are convenience, not required
130
135
  }
131
136
  }
137
+
138
+ try { fs.unlinkSync(ps1Path); } catch {}
132
139
  }
133
140
 
134
141
  function uninstallWindows() {
package/watcher.js CHANGED
@@ -159,31 +159,51 @@ function extractNewUsage(filePath) {
159
159
  }
160
160
 
161
161
  if (!msg) {
162
- // ── Codex CLI format ──────────────────────────────────────────
163
- // Codex events have { type: "event", payload: { type: "token_count", ... } }
164
- // with cumulative input_tokens, output_tokens, reasoning_tokens, cached_input_tokens
165
- if (obj.type === 'event' && obj.payload?.type === 'token_count') {
166
- const p = obj.payload;
162
+ // ── Codex CLI format (old: type="event", new: type="event_msg") ──
163
+ // Old format: { type: "event", payload: { type: "token_count", input_tokens, ... } }
164
+ // New format: { type: "event_msg", payload: { type: "token_count", info: { total_token_usage: { input_tokens, ... } } } }
165
+ if ((obj.type === 'event' || obj.type === 'event_msg') && obj.payload?.type === 'token_count') {
166
+ // Extract token counts — handle both old (flat) and new (nested) format
167
+ let inputTokens, outputTokens, reasoningTokens, cachedTokens;
168
+
169
+ if (obj.payload.info?.total_token_usage) {
170
+ // New format (2026+): cumulative totals in info.total_token_usage
171
+ const t = obj.payload.info.total_token_usage;
172
+ inputTokens = t.input_tokens || 0;
173
+ outputTokens = t.output_tokens || 0;
174
+ reasoningTokens = t.reasoning_output_tokens || t.reasoning_tokens || 0;
175
+ cachedTokens = t.cached_input_tokens || 0;
176
+ } else {
177
+ // Old format: flat fields on payload
178
+ inputTokens = obj.payload.input_tokens || 0;
179
+ outputTokens = obj.payload.output_tokens || 0;
180
+ reasoningTokens = obj.payload.reasoning_tokens || 0;
181
+ cachedTokens = obj.payload.cached_input_tokens || 0;
182
+ }
183
+
184
+ // Skip events with no token info (e.g. rate limit only events)
185
+ if (inputTokens === 0 && outputTokens === 0) continue;
186
+
167
187
  const codexModel = obj.turn_context?.model || 'codex';
188
+
168
189
  // Token counts are cumulative per session; we store deltas
169
- // Use a per-file tracker for the previous cumulative values
170
190
  const prevKey = filePath;
171
191
  const prev = _codexCumulative[prevKey] || { input: 0, output: 0, reasoning: 0, cached: 0 };
172
- const deltaInput = (p.input_tokens || 0) - prev.input;
173
- const deltaOutput = (p.output_tokens || 0) - prev.output;
174
- const deltaReasoning = (p.reasoning_tokens || 0) - prev.reasoning;
192
+ const deltaInput = inputTokens - prev.input;
193
+ const deltaOutput = outputTokens - prev.output;
194
+ const deltaReasoning = reasoningTokens - prev.reasoning;
175
195
 
176
196
  _codexCumulative[prevKey] = {
177
- input: p.input_tokens || 0,
178
- output: p.output_tokens || 0,
179
- reasoning: p.reasoning_tokens || 0,
180
- cached: p.cached_input_tokens || 0,
197
+ input: inputTokens,
198
+ output: outputTokens,
199
+ reasoning: reasoningTokens,
200
+ cached: cachedTokens,
181
201
  };
182
202
 
183
203
  // Skip if no new tokens (duplicate or first read)
184
204
  if (deltaInput <= 0 && deltaOutput <= 0) continue;
185
205
 
186
- const hashKey = `${filePath}:${lineOffset}:codex:${p.input_tokens}:${p.output_tokens}`;
206
+ const hashKey = `${filePath}:${lineOffset}:codex:${inputTokens}:${outputTokens}`;
187
207
  const hash = crypto.createHash('md5').update(hashKey).digest('hex');
188
208
  if (isDuplicate(hash)) continue;
189
209
 
@@ -194,7 +214,7 @@ function extractNewUsage(filePath) {
194
214
  inputTokens: deltaInput,
195
215
  outputTokens: deltaOutput,
196
216
  thinkingTokens: deltaReasoning,
197
- cacheReadTokens: 0,
217
+ cacheReadTokens: cachedTokens,
198
218
  cacheWriteTokens: 0,
199
219
  });
200
220
  continue;