@winston.wan/burn-your-money 2.0.5 → 2.1.1

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.
package/src/statusline.sh CHANGED
@@ -1,213 +1,250 @@
1
- #!/bin/bash
2
- # Burn Your Money - Claude Code 状态栏脚本
3
- # 格式:[模型] 今日:token 费用 🔥速度 | 总计:token 费用
4
-
5
- # 配置文件
6
- CONFIG_FILE="$HOME/.claude/burn-your-money-config.json"
7
- SESSION_STATE="$HOME/.claude/cache/session-state.json"
8
-
9
- # 初始化配置
10
- init_config() {
11
- if [ ! -f "$CONFIG_FILE" ]; then
12
- jq -n '{
13
- theme: "fire",
14
- alert_daily: 10.0,
15
- alert_weekly: 50.0,
16
- show_burn_rate: true,
17
- show_eta: true,
18
- show_trend: false
19
- }' > "$CONFIG_FILE"
20
- fi
21
- }
22
-
23
- # 读取配置
24
- # 读取配置
25
- init_config
26
- if [ -f "$CONFIG_FILE" ]; then
27
- THEME=$(cat "$CONFIG_FILE" | jq -r '.theme // "fire"' 2>/dev/null)
28
- ALERT_DAILY=$(cat "$CONFIG_FILE" | jq -r '.alert_daily // 10' 2>/dev/null)
29
- SHOW_BURN_RATE=$(cat "$CONFIG_FILE" | jq -r '.show_burn_rate // true' 2>/dev/null)
30
- SHOW_ETA=$(cat "$CONFIG_FILE" | jq -r '.show_eta // true' 2>/dev/null)
31
- else
32
- THEME="fire"
33
- ALERT_DAILY=10
34
- SHOW_BURN_RATE=true
35
- SHOW_ETA=true
36
- fi
37
-
38
- # 主题颜色定义
39
- case "$THEME" in
40
- fire)
41
- TODAY_COLOR="0;31" # 红色
42
- COST_COLOR="0;31" # 红色
43
- BURN_COLOR="0;31" # 红色
44
- ;;
45
- ocean)
46
- TODAY_COLOR="0;34" # 蓝色
47
- COST_COLOR="0;36" # 青色
48
- BURN_COLOR="0;36" # 青色
49
- ;;
50
- forest)
51
- TODAY_COLOR="0;32" # 绿色
52
- COST_COLOR="0;33" # 黄色
53
- BURN_COLOR="0;32" # 绿色
54
- ;;
55
- golden)
56
- TODAY_COLOR="0;33" # 黄色
57
- COST_COLOR="0;33" # 黄色
58
- BURN_COLOR="1;33" # 亮黄色
59
- ;;
60
- *)
61
- TODAY_COLOR="0;36" # 默认青色
62
- COST_COLOR="0;34" # 默认蓝色
63
- BURN_COLOR="0;31" # 默认红色
64
- ;;
65
- esac
66
-
67
- # 格式化数字(添加 K/M 后缀)
68
- format_number() {
69
- local num=$1
70
- if [ "$num" -ge 1000000000 ]; then
71
- awk "BEGIN {printf \"%.1fB\", $num/1000000000}"
72
- elif [ "$num" -ge 1000000 ]; then
73
- awk "BEGIN {printf \"%.1fM\", $num/1000000}"
74
- elif [ "$num" -ge 1000 ]; then
75
- awk "BEGIN {printf \"%.1fK\", $num/1000}"
76
- else
77
- echo "$num"
78
- fi
79
- }
80
-
81
- # 格式化费用
82
- format_cost() {
83
- local cost=$1
84
- awk "BEGIN {printf \"\$%.2f\", $cost}"
85
- }
86
-
87
- # 格式化燃烧速度
88
- format_burn_rate() {
89
- local rate=$1
90
- if [ "$rate" -ge 1000 ]; then
91
- awk "BEGIN {printf \"%.1fK tok/s\", $rate/1000}"
92
- else
93
- awk "BEGIN {printf \"%.0f tok/s\", $rate}"
94
- fi
95
- }
96
-
97
- # 获取或创建会话状态
98
- get_session_start_time() {
99
- local session_id=$(cat "$TEMP_JSON" | jq -r '.session_id // "default"' 2>/dev/null || echo "default")
100
- local now=$(date +%s)
101
-
102
- if [ -f "$SESSION_STATE" ]; then
103
- local cached_id=$(cat "$SESSION_STATE" | jq -r '.session_id // ""' 2>/dev/null || echo "")
104
- local cached_time=$(cat "$SESSION_STATE" | jq -r '.start_time // 0' 2>/dev/null || echo "0")
105
-
106
- if [ "$cached_id" = "$session_id" ] && [ "$cached_time" -gt 0 ]; then
107
- echo "$cached_time"
108
- return
109
- fi
110
- fi
111
-
112
- # 新会话,记录开始时间
113
- echo "$now"
114
- jq -n --arg id "$session_id" --arg time "$now" '{session_id: $id, start_time: ($time | tonumber)}' > "$SESSION_STATE"
115
- echo "$now"
116
- }
117
-
118
- # 主逻辑:使用临时文件存储 stdin
119
- TEMP_JSON=$(mktemp)
120
- trap "rm -f $TEMP_JSON" EXIT
121
- cat > "$TEMP_JSON"
122
-
123
- # 读取模型名称
124
- model=$(cat "$TEMP_JSON" | jq -r '.model.display_name // "Unknown"' 2>/dev/null || echo "Unknown")
125
-
126
- # 读取当前会话的 token 数据(从 context_window)
127
- current_input=$(cat "$TEMP_JSON" | jq -r '.context_window.total_input_tokens // 0' 2>/dev/null || echo "0")
128
- current_output=$(cat "$TEMP_JSON" | jq -r '.context_window.total_output_tokens // 0' 2>/dev/null || echo "0")
129
- current_session_tokens=$((current_input + current_output))
130
-
131
- # 计算燃烧速度
132
- session_start=$(get_session_start_time)
133
- current_time=$(date +%s)
134
- elapsed=$((current_time - session_start))
135
-
136
- if [ "$elapsed" -gt 0 ] && [ "$current_session_tokens" -gt 0 ]; then
137
- burn_rate=$(awk "BEGIN {printf \"%.0f\", ${current_session_tokens} / ${elapsed}}")
138
- else
139
- burn_rate=0
140
- fi
141
-
142
- # 获取历史数据(缓存 30 秒)
143
- history_cache="$HOME/.claude/cache/history-cache.json"
144
- now=$(date +%s)
145
- cache_age=999999
146
- if [ -f "$history_cache" ]; then
147
- cache_time=$(stat -c %Y "$history_cache" 2>/dev/null || stat -f %m "$history_cache" 2>/dev/null || echo "0")
148
- cache_age=$((now - cache_time))
149
- fi
150
-
151
- if [ $cache_age -gt 30 ]; then
152
- ~/.claude/scripts/token-history.sh summary > "$history_cache" 2>/dev/null || echo '{}' > "$history_cache"
153
- fi
154
-
155
- # 读取历史数据并加上当前会话
156
- history_today_tokens=$(jq -r '.today_tokens // 0' "$history_cache" 2>/dev/null || echo "0")
157
- history_today_cost=$(jq -r '.today_cost // 0' "$history_cache" 2>/dev/null || echo "0")
158
-
159
- # 今日数据 = 历史数据 + 当前会话
160
- today_tokens=$((history_today_tokens + current_session_tokens))
161
- # 简单估算今日费用(按比例)
162
- if [ "$history_today_tokens" -gt 0 ] && [ "$history_today_cost" != "0" ]; then
163
- today_cost=$(awk "BEGIN {printf \"%.2f\", ${history_today_cost} / ${history_today_tokens} * ${today_tokens}}")
164
- elif [ "$today_tokens" -gt 0 ]; then
165
- # 如果没有历史费用数据,用平均价格估算
166
- today_cost=$(awk "BEGIN {printf \"%.2f\", ${today_tokens} * 0.000006}")
167
- else
168
- today_cost="0.00"
169
- fi
170
-
171
- total_tokens_all=$(jq -r '.total_tokens_all // 0' "$history_cache" 2>/dev/null || echo "0")
172
- total_cost=$(jq -r '.week_cost // 0' "$history_cache" 2>/dev/null || echo "0")
173
-
174
- # 检查费用警报
175
- alert_needed=false
176
- if awk "BEGIN {exit !($today_cost >= $ALERT_DAILY)}"; then
177
- alert_needed=true
178
- fi
179
-
180
- # 构建输出
181
- output=""
182
-
183
- # 模型名称
184
- if [ "$alert_needed" = true ]; then
185
- output+="\\033[1;31m[$model]\\033[0m " # 红色闪烁
186
- else
187
- output+="\\033[0;90m[$model]\\033[0m "
188
- fi
189
-
190
- # 今日数据
191
- today_display=$(format_number $today_tokens)
192
- today_cost_display=$(format_cost $today_cost)
193
- if [ "$today_tokens" -gt 0 ] || awk "BEGIN {exit !($today_cost > 0)}"; then
194
- output+="\\033[${TODAY_COLOR}m今日:${today_display}\\033[0m "
195
- output+="\\033[${COST_COLOR}m${today_cost_display}\\033[0m "
196
- else
197
- output+="\\033[0;90m今日:0\\033[0m "
198
- output+="\\033[0;90m\$0.00\\033[0m "
199
- fi
200
-
201
- # 燃烧速度
202
- if [ "$SHOW_BURN_RATE" = true ] && [ "$burn_rate" -gt 0 ]; then
203
- burn_display=$(format_burn_rate $burn_rate)
204
- output+="\\033[${BURN_COLOR}m🔥${burn_display}\\033[0m "
205
- fi
206
-
207
- # 总计数据
208
- total_display=$(format_number $total_tokens_all)
209
- total_cost_display=$(format_cost $total_cost)
210
- output+="| \\033[0;90m总计:${total_display}\\033[0m "
211
- output+="\\033[0;90m${total_cost_display}\\033[0m "
212
-
213
- printf "%b" "$output"
1
+ #!/bin/bash
2
+ # Burn Your Money - Claude Code 状态栏脚本
3
+ # 格式:[模型] 今日:token 费用 🔥速度 | 总计:token 费用
4
+
5
+ # 设置 jq 命令路径(支持自定义安装的 jq)
6
+ if [ -n "$JQ_PATH" ]; then
7
+ JQ_CMD="$JQ_PATH"
8
+ elif [ -f "$HOME/.claude/bin/jq.exe" ]; then
9
+ JQ_CMD="$HOME/.claude/bin/jq.exe"
10
+ elif [ -f "$HOME/.claude/bin/jq" ]; then
11
+ JQ_CMD="$HOME/.claude/bin/jq"
12
+ else
13
+ JQ_CMD="jq"
14
+ fi
15
+
16
+ # 验证 jq 是否可用
17
+ if ! command -v "$JQ_CMD" >/dev/null 2>&1; then
18
+ # jq 不可用,返回简单错误信息
19
+ printf "[BurnYourMoney] Error: jq not found. Run: npm install -g @winston.wan/burn-your-money"
20
+ exit 0
21
+ fi
22
+
23
+ # 配置文件
24
+ CONFIG_FILE="$HOME/.claude/burn-your-money-config.json"
25
+ SESSION_STATE="$HOME/.claude/cache/session-state.json"
26
+ TODAY_STATE="$HOME/.claude/cache/today-state.json"
27
+ HISTORY_CACHE="$HOME/.claude/cache/history-cache.json"
28
+
29
+ # 初始化配置
30
+ init_config() {
31
+ if [ ! -f "$CONFIG_FILE" ]; then
32
+ $JQ_CMD -n '{
33
+ theme: "fire",
34
+ alert_daily: 10.0,
35
+ alert_weekly: 50.0,
36
+ show_burn_rate: true,
37
+ show_eta: true,
38
+ show_trend: false
39
+ }' > "$CONFIG_FILE"
40
+ fi
41
+ }
42
+
43
+ # 1. 批量读取配置 (One JQ pass)
44
+ init_config
45
+ if [ -f "$CONFIG_FILE" ]; then
46
+ # 使用 eval 批量赋值变量
47
+ eval $("$JQ_CMD" -r '@sh "THEME=\(.theme // "fire") ALERT_DAILY=\(.alert_daily // 10) SHOW_BURN_RATE=\(.show_burn_rate // true)"' "$CONFIG_FILE" 2>/dev/null)
48
+ else
49
+ THEME="fire"
50
+ ALERT_DAILY=10
51
+ SHOW_BURN_RATE=true
52
+ fi
53
+
54
+ # 主题颜色定义
55
+ case "$THEME" in
56
+ fire) TODAY_COLOR="0;31"; COST_COLOR="0;31"; BURN_COLOR="0;31" ;;
57
+ ocean) TODAY_COLOR="0;34"; COST_COLOR="0;36"; BURN_COLOR="0;36" ;;
58
+ forest) TODAY_COLOR="0;32"; COST_COLOR="0;33"; BURN_COLOR="0;32" ;;
59
+ golden) TODAY_COLOR="0;33"; COST_COLOR="0;33"; BURN_COLOR="1;33" ;;
60
+ *) TODAY_COLOR="0;36"; COST_COLOR="0;34"; BURN_COLOR="0;31" ;;
61
+ esac
62
+
63
+ # 格式化函数 (纯 Bash/Awk 实现,无外部进程调用)
64
+ format_number() {
65
+ local num=$1
66
+ if [ "$num" -ge 1000000000 ]; then awk "BEGIN {printf \"%.1fB\", $num/1000000000}";
67
+ elif [ "$num" -ge 1000000 ]; then awk "BEGIN {printf \"%.1fM\", $num/1000000}";
68
+ elif [ "$num" -ge 1000 ]; then awk "BEGIN {printf \"%.1fK\", $num/1000}";
69
+ else echo "$num"; fi
70
+ }
71
+
72
+ format_cost() {
73
+ awk "BEGIN {printf \"\$%.2f\", $1}"
74
+ }
75
+
76
+ format_burn_rate() {
77
+ local rate=$1
78
+ if [ "$rate" -ge 1000 ]; then awk "BEGIN {printf \"%.1fK tok/s\", $rate/1000}";
79
+ else awk "BEGIN {printf \"%.0f tok/s\", $rate}"; fi
80
+ }
81
+
82
+ # 2. 核心数据解析 (One JQ pass from stdin)
83
+ # 将 stdin 存入临时文件(避免大 JSON 管道卡顿)
84
+ TEMP_JSON=$(mktemp)
85
+ trap "rm -f $TEMP_JSON" EXIT
86
+ cat > "$TEMP_JSON"
87
+
88
+ # 一次性提取所有需要的字段
89
+ # 注意:使用 @sh 确保字符串安全引用
90
+ eval $("$JQ_CMD" -r '
91
+ @sh "model_name=\(.model.display_name // "Unknown")
92
+ session_id=\(.session_id // "default")
93
+ input=\(.context_window.total_input_tokens // 0)
94
+ output=\(.context_window.total_output_tokens // 0)
95
+ cache=\(.context_window.current_usage.cache_read_input_tokens // 0)
96
+ current_cost=\(.cost.total_cost_usd // 0)"
97
+ ' "$TEMP_JSON" 2>/dev/null)
98
+
99
+ # 计算当前会话 Token 总量
100
+ current_session_tokens=$((input + output + cache))
101
+
102
+ # 3. 燃烧速度计算 (Burn Rate)
103
+ current_time=$(date +%s)
104
+ start_time_cache=0
105
+
106
+ # 读取 Session State (尽可能不做写入,除非是新 Session)
107
+ if [ -f "$SESSION_STATE" ]; then
108
+ eval $("$JQ_CMD" -r '@sh "cached_id=\(.session_id // "") cached_time=\(.start_time // 0)"' "$SESSION_STATE" 2>/dev/null)
109
+ else
110
+ cached_id=""
111
+ cached_time=0
112
+ fi
113
+
114
+ if [ "$cached_id" = "$session_id" ] && [ "$cached_time" -ne 0 ]; then
115
+ start_time_cache=$cached_time
116
+ else
117
+ # New session detected - 使用原子写入
118
+ start_time_cache=$current_time
119
+ SESSION_STATE_TMP="${SESSION_STATE}.tmp.$$"
120
+ $JQ_CMD -n --arg id "$session_id" --arg time "$current_time" '{session_id: $id, start_time: ($time | tonumber)}' > "$SESSION_STATE_TMP" 2>/dev/null && mv "$SESSION_STATE_TMP" "$SESSION_STATE"
121
+ fi
122
+
123
+ elapsed=$((current_time - start_time_cache))
124
+ burn_rate=0
125
+ if [ "$elapsed" -gt 0 ] && [ "$current_session_tokens" -gt 0 ]; then
126
+ burn_rate=$(awk "BEGIN {printf \"%.0f\", ${current_session_tokens} / ${elapsed}}")
127
+ fi
128
+
129
+ # 4. 历史数据缓存 ( 30 秒刷新)
130
+ # 确保 cache 目录存在
131
+ mkdir -p "$(dirname "$HISTORY_CACHE")" 2>/dev/null
132
+
133
+ now=$(date +%s)
134
+ cache_age=999999
135
+ if [ -f "$HISTORY_CACHE" ]; then
136
+ # Portable modification time check
137
+ if [ "$(uname)" = "Darwin" ]; then
138
+ cache_time=$(stat -f %m "$HISTORY_CACHE" 2>/dev/null || echo 0)
139
+ else
140
+ cache_time=$(stat -c %Y "$HISTORY_CACHE" 2>/dev/null || echo 0)
141
+ fi
142
+ cache_age=$((now - cache_time))
143
+ else
144
+ echo '{}' > "$HISTORY_CACHE"
145
+ fi
146
+
147
+ # 如果缓存太旧,并在后台刷新(虽然后台刷新可能在 statusline 不合适,但我们可以同步调用历史脚本)
148
+ # 为了性能,这里我们仍然做同步调用,但减少频率
149
+ if [ $cache_age -gt 30 ]; then
150
+ # 重新生成历史数据
151
+ ~/.claude/scripts/token-history.sh summary > "$HISTORY_CACHE" 2>/dev/null
152
+ fi
153
+
154
+ # 读取历史总计 (One pass)
155
+ eval $("$JQ_CMD" -r '@sh "history_total_tokens=\(.total_tokens_all // 0) history_total_cost=\(.total_cost // 0)"' "$HISTORY_CACHE" 2>/dev/null)
156
+
157
+ # 5. 今日数据持久化 (One-pass Read & Write)
158
+ # 逻辑:读取旧状态 -> 计算差值 -> 写入新状态
159
+ today_date=$(date +%Y-%m-%d)
160
+
161
+ if [ -f "$TODAY_STATE" ]; then
162
+ eval $("$JQ_CMD" -r '@sh "saved_date=\(.date // "") saved_tokens=\(.tokens // 0) saved_cost=\(.cost // 0) last_session_tokens=\(.last_session_tokens // 0) last_session_cost=\(.last_session_cost // 0)"' "$TODAY_STATE" 2>/dev/null)
163
+ else
164
+ saved_date=""
165
+ saved_tokens=0
166
+ saved_cost=0
167
+ last_session_tokens=0
168
+ last_session_cost=0
169
+ fi
170
+
171
+ if [ "$saved_date" != "$today_date" ]; then
172
+ # 新的一天,重置累计数据
173
+ saved_tokens=0
174
+ saved_cost=0
175
+ last_session_tokens=0
176
+ last_session_cost=0
177
+ fi
178
+
179
+ # 计算增量
180
+ token_increment=$((current_session_tokens - last_session_tokens))
181
+ if [ "$token_increment" -lt 0 ]; then token_increment=0; fi # 防御性编程
182
+
183
+ # 浮点数增量计算
184
+ cost_increment=$(awk "BEGIN {print ($current_cost - $last_session_cost < 0 ? 0 : $current_cost - $last_session_cost)}")
185
+
186
+ # 计算新的今日累计
187
+ new_today_tokens=$((saved_tokens + token_increment))
188
+ new_today_cost=$(awk "BEGIN {printf \"%.5f\", $saved_cost + $cost_increment}")
189
+
190
+ # 写入新状态 (一次性写入,使用原子写入避免并发问题)
191
+ # 先写入临时文件,然后重命名,确保原子性
192
+ TODAY_STATE_TMP="${TODAY_STATE}.tmp.$$"
193
+ $JQ_CMD -n \
194
+ --arg date "$today_date" \
195
+ --arg tokens "$new_today_tokens" \
196
+ --arg cost "$new_today_cost" \
197
+ --arg last_sess_tok "$current_session_tokens" \
198
+ --arg last_sess_cost "$current_cost" \
199
+ '{
200
+ date: $date,
201
+ tokens: ($tokens|tonumber),
202
+ cost: ($cost|tonumber),
203
+ last_session_tokens: ($last_sess_tok|tonumber),
204
+ last_session_cost: ($last_sess_cost|tonumber)
205
+ }' > "$TODAY_STATE_TMP" 2>/dev/null && mv "$TODAY_STATE_TMP" "$TODAY_STATE"
206
+
207
+ # 6. 最终计算与显示
208
+ total_tokens_all=$((history_total_tokens + current_session_tokens))
209
+ total_cost_all=$(awk "BEGIN {print $history_total_cost + $current_cost}")
210
+
211
+ # 警报检查
212
+ is_alert=false
213
+ if awk "BEGIN {exit !($new_today_cost >= $ALERT_DAILY)}"; then
214
+ is_alert=true
215
+ fi
216
+
217
+ # 构建输出字符串
218
+ output=""
219
+
220
+ # [Model]
221
+ if [ "$is_alert" = true ]; then
222
+ output+="\\033[1;31m[${model_name}]\\033[0m "
223
+ else
224
+ output+="\\033[0;90m[${model_name}]\\033[0m "
225
+ fi
226
+
227
+ # Today
228
+ d_today_tokens=$(format_number $new_today_tokens)
229
+ d_today_cost=$(format_cost $new_today_cost)
230
+
231
+ if [ "$new_today_tokens" -gt 0 ]; then
232
+ output+="\\033[${TODAY_COLOR}m今日:${d_today_tokens}\\033[0m "
233
+ output+="\\033[${COST_COLOR}m${d_today_cost}\\033[0m "
234
+ else
235
+ output+="\\033[0;90m今日:0\\033[0m \\033[0;90m\$0.00\\033[0m "
236
+ fi
237
+
238
+ # Burn Rate
239
+ if [ "$SHOW_BURN_RATE" = "true" ] && [ "$burn_rate" -gt 0 ]; then
240
+ d_burn=$(format_burn_rate $burn_rate)
241
+ output+="\\033[${BURN_COLOR}m🔥${d_burn}\\033[0m "
242
+ fi
243
+
244
+ # Total
245
+ d_total_tokens=$(format_number $total_tokens_all)
246
+ d_total_cost=$(format_cost $total_cost_all)
247
+ output+="| \\033[0;90m总计:${d_total_tokens}\\033[0m \\033[0;90m${d_total_cost}\\033[0m "
248
+
249
+ # Print final result
250
+ printf "%b" "$output"