remark_vars_test.go 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. package sub
  2. import (
  3. "strings"
  4. "testing"
  5. "github.com/mhsanaei/3x-ui/v3/internal/database/model"
  6. "github.com/mhsanaei/3x-ui/v3/internal/xray"
  7. )
  8. const gb = int64(1024 * 1024 * 1024)
  9. // expandCtx builds a remarkContext from explicit pieces for token tests.
  10. func expandCtx(client model.Client, stats xray.ClientTraffic, inbound *model.Inbound) remarkContext {
  11. return remarkContext{client: client, stats: stats, inbound: inbound}
  12. }
  13. func TestExpandRemarkVars(t *testing.T) {
  14. inbound := &model.Inbound{Remark: "Germany"}
  15. client := model.Client{
  16. Email: "[email protected]",
  17. ID: "3f2a9c1b-aaaa-bbbb-cccc-1234567890ab",
  18. TgID: 123456789,
  19. SubID: "subABC",
  20. Comment: "vip",
  21. Reset: 30,
  22. CreatedAt: 1_700_000_000_000,
  23. }
  24. // 50GB total, 8GB used (5 up + 3 down), enabled, no expiry.
  25. stats := xray.ClientTraffic{
  26. Enable: true,
  27. Total: 50 * gb,
  28. Up: 5 * gb,
  29. Down: 3 * gb,
  30. }
  31. ctx := expandCtx(client, stats, inbound)
  32. cases := []struct{ tmpl, want string }{
  33. {"{{EMAIL}}", "[email protected]"},
  34. {"{{INBOUND}}", "Germany"}, // no host remark in ctx → inbound remark
  35. {"{{HOST}}", ""}, // no host remark in ctx → empty
  36. {"{{ID}}", client.ID},
  37. {"{{SHORT_ID}}", "3f2a9c1b"},
  38. {"{{TELEGRAM_ID}}", "123456789"},
  39. {"{{SUB_ID}}", "subABC"},
  40. {"{{COMMENT}}", "vip"},
  41. {"{{RESET_DAYS}}", "30"},
  42. {"{{CREATED_UNIX}}", "1700000000"},
  43. {"{{TRAFFIC_USED}}", "8.00GB"},
  44. {"{{TRAFFIC_LEFT}}", "42.00GB"},
  45. {"{{TRAFFIC_TOTAL}}", "50.00GB"},
  46. {"{{TRAFFIC_USED_BYTES}}", "8589934592"},
  47. {"{{TRAFFIC_TOTAL_BYTES}}", "53687091200"},
  48. {"{{UP}}", "5.00GB"},
  49. {"{{DOWN}}", "3.00GB"},
  50. {"{{STATUS}}", "active"},
  51. {"{{EXPIRE_UNIX}}", "0"}, // no expiry
  52. {"{{EXPIRE_DATE}}", ""}, // no fixed date
  53. {"{{UNKNOWN_TOKEN}}", ""}, // unknown → empty, never literal
  54. {"DE {{EMAIL}} ok", "DE [email protected] ok"},
  55. {"{{EMAIL}}-{{SHORT_ID}}", "[email protected]"},
  56. {"no tokens here", "no tokens here"},
  57. }
  58. for _, c := range cases {
  59. if got := expandRemarkVars(c.tmpl, ctx); got != c.want {
  60. t.Errorf("expandRemarkVars(%q) = %q, want %q", c.tmpl, got, c.want)
  61. }
  62. }
  63. // The unlimited tokens still render ∞ at the value layer; expandRemarkVars
  64. // is what drops an all-unlimited segment (see TestExpandRemarkVars_DropUnlimitedSegments).
  65. if got := remarkVarValue("DAYS_LEFT", ctx); got != "∞" {
  66. t.Errorf("remarkVarValue(DAYS_LEFT) = %q, want ∞", got)
  67. }
  68. }
  69. func TestExpandRemarkVars_EdgeCases(t *testing.T) {
  70. // Unlimited total → ∞ for human forms, 0 bytes for *_BYTES left. Checked at
  71. // the value layer: expandRemarkVars would drop a bare ∞ segment.
  72. unlimited := expandCtx(model.Client{}, xray.ClientTraffic{Enable: true, Total: 0, Up: gb}, nil)
  73. if got := remarkVarValue("TRAFFIC_TOTAL", unlimited); got != "∞" {
  74. t.Errorf("unlimited TRAFFIC_TOTAL = %q, want ∞", got)
  75. }
  76. if got := remarkVarValue("TRAFFIC_LEFT", unlimited); got != "∞" {
  77. t.Errorf("unlimited TRAFFIC_LEFT = %q, want ∞", got)
  78. }
  79. if got := expandRemarkVars("{{TRAFFIC_LEFT_BYTES}}", unlimited); got != "0" {
  80. t.Errorf("unlimited TRAFFIC_LEFT_BYTES = %q, want 0", got)
  81. }
  82. // TgID zero → empty.
  83. if got := expandRemarkVars("{{TELEGRAM_ID}}", unlimited); got != "" {
  84. t.Errorf("zero TgID = %q, want empty", got)
  85. }
  86. // Over-quota usage clamps left to 0, not negative.
  87. over := expandCtx(model.Client{}, xray.ClientTraffic{Enable: true, Total: gb, Up: 2 * gb}, nil)
  88. if got := expandRemarkVars("{{TRAFFIC_LEFT_BYTES}}", over); got != "0" {
  89. t.Errorf("over-quota TRAFFIC_LEFT_BYTES = %q, want 0", got)
  90. }
  91. // Delayed-start (negative expiry) gives deterministic whole days.
  92. delayed := expandCtx(model.Client{}, xray.ClientTraffic{Enable: true, ExpiryTime: -864_000_000}, nil)
  93. if got := expandRemarkVars("{{DAYS_LEFT}}", delayed); got != "10" {
  94. t.Errorf("delayed-start DAYS_LEFT = %q, want 10", got)
  95. }
  96. }
  97. // An unlimited client drops the quota/expiry segments whole — decoration and the
  98. // "|" separator included — instead of printing "📊∞|⏳∞D".
  99. func TestExpandRemarkVars_DropUnlimitedSegments(t *testing.T) {
  100. const tmpl = "{{INBOUND}}|📊{{TRAFFIC_LEFT}}|⏳{{DAYS_LEFT}}D"
  101. inbound := &model.Inbound{Remark: "host"}
  102. // No limit at all → only the name segment survives.
  103. unlimited := expandCtx(model.Client{}, xray.ClientTraffic{Enable: true}, inbound)
  104. if got := expandRemarkVars(tmpl, unlimited); got != "host" {
  105. t.Errorf("fully unlimited = %q, want %q", got, "host")
  106. }
  107. // Limited traffic but no expiry → traffic stays, the expiry segment drops.
  108. noExpiry := expandCtx(model.Client{}, xray.ClientTraffic{Enable: true, Total: 50 * gb, Up: 8 * gb}, inbound)
  109. if got := expandRemarkVars(tmpl, noExpiry); got != "host|📊42.00GB" {
  110. t.Errorf("no-expiry = %q, want %q", got, "host|📊42.00GB")
  111. }
  112. // A segment mixing an unlimited token with another value is kept whole,
  113. // decoration and ∞ included — only all-unlimited segments drop.
  114. mixed := expandCtx(model.Client{Email: "john"}, xray.ClientTraffic{Enable: true}, inbound)
  115. if got := expandRemarkVars("{{EMAIL}} 📊{{TRAFFIC_LEFT}}", mixed); got != "john 📊∞" {
  116. t.Errorf("mixed segment = %q, want %q", got, "john 📊∞")
  117. }
  118. }
  119. func TestClientStatus(t *testing.T) {
  120. cases := []struct {
  121. name string
  122. st xray.ClientTraffic
  123. want string
  124. }{
  125. {"disabled", xray.ClientTraffic{Enable: false}, "disabled"},
  126. {"active", xray.ClientTraffic{Enable: true}, "active"},
  127. {"expired", xray.ClientTraffic{Enable: true, ExpiryTime: 1000}, "expired"}, // 1s past epoch
  128. {"depleted", xray.ClientTraffic{Enable: true, Total: gb, Up: gb}, "depleted"},
  129. }
  130. for _, c := range cases {
  131. if got := clientStatus(c.st); got != c.want {
  132. t.Errorf("%s: clientStatus = %q, want %q", c.name, got, c.want)
  133. }
  134. }
  135. }
  136. // hostRemarkService builds a SubService + inbound + client/stats for remark tests.
  137. func hostRemarkService(template string) (*SubService, *model.Inbound, model.Client) {
  138. s := &SubService{remarkTemplate: template, subscriptionBody: true}
  139. inbound := &model.Inbound{
  140. Remark: "DE",
  141. ClientStats: []xray.ClientTraffic{{
  142. Email: "[email protected]",
  143. Enable: true,
  144. Total: 100 * gb,
  145. Up: 15 * gb,
  146. Down: 5 * gb,
  147. ExpiryTime: -864_000_000, // delayed-start: deterministic 10 days
  148. }},
  149. }
  150. client := model.Client{Email: "[email protected]"}
  151. return s, inbound, client
  152. }
  153. // The config name is always the inbound's own remark; the host endpoint's remark
  154. // never substitutes it (it is reachable only through {{HOST}}).
  155. func TestGenHostRemark_ConfigNameUsesInbound(t *testing.T) {
  156. s, inbound, client := hostRemarkService("") // no template → config name only
  157. if got := s.genHostRemark(inbound, client, "Relay", ""); got != "DE" {
  158. t.Fatalf("genHostRemark = %q, want %q (inbound remark, host ignored)", got, "DE")
  159. }
  160. if got := s.genHostRemark(inbound, client, "", ""); got != "DE" {
  161. t.Fatalf("genHostRemark (no host remark) = %q, want %q", got, "DE")
  162. }
  163. }
  164. // In the body the template applies: {{INBOUND}} is always the inbound's remark
  165. // and {{HOST}} the host's own remark, so the two can be shown side by side.
  166. func TestGenHostRemark_GlobalTemplate(t *testing.T) {
  167. // {{INBOUND}} resolves to the inbound remark regardless of the host remark.
  168. s, inbound, client := hostRemarkService("{{INBOUND}} | {{TRAFFIC_LEFT}} | {{DAYS_LEFT}}d")
  169. if got := s.genHostRemark(inbound, client, "CDN", ""); got != "DE | 80.00GB | 10d" {
  170. t.Fatalf("global template ({{INBOUND}} = inbound) = %q", got)
  171. }
  172. // {{INBOUND}} and {{HOST}} side by side show both, distinctly (#5443).
  173. s2, inbound2, client2 := hostRemarkService("{{INBOUND}}|{{HOST}}|{{TRAFFIC_LEFT}}")
  174. if got := s2.genHostRemark(inbound2, client2, "CDN", ""); got != "DE|CDN|80.00GB" {
  175. t.Fatalf("global template (inbound + host) = %q, want %q", got, "DE|CDN|80.00GB")
  176. }
  177. // {{HOST}} is the host's own remark even when the inbound has one of its own.
  178. s3, inbound3, client3 := hostRemarkService("{{HOST}}")
  179. if got := s3.genHostRemark(inbound3, client3, "CDN", ""); got != "CDN" {
  180. t.Fatalf("{{HOST}} token = %q, want CDN", got)
  181. }
  182. }
  183. // A global template also drives non-host links via genRemark; {{HOST}} = the
  184. // legacy externalProxy remark passed as extra.
  185. func TestGenRemark_GlobalTemplate(t *testing.T) {
  186. s, inbound, _ := hostRemarkService("{{EMAIL}} | {{TRAFFIC_LEFT}}")
  187. got := s.genRemark(inbound, "[email protected]", "", "")
  188. if got != "[email protected] | 80.00GB" {
  189. t.Fatalf("global template (non-host) = %q", got)
  190. }
  191. }
  192. // With no template, genRemark composes the fallback model and adds no suffix.
  193. func TestGenRemark_NoTemplate_NoSuffix(t *testing.T) {
  194. s, inbound, _ := hostRemarkService("")
  195. got := s.genRemark(inbound, "[email protected]", "Relay", "")
  196. if got != "DE-Relay" {
  197. t.Fatalf("genRemark = %q, want %q (no suffix)", got, "DE-Relay")
  198. }
  199. }
  200. // The per-client info part of the template renders only on a client's first
  201. // link of the request; later links show the name-only template.
  202. func TestUsageOnFirstLinkOnly(t *testing.T) {
  203. s, inbound, client := hostRemarkService("{{INBOUND}}|📊{{TRAFFIC_LEFT}}|⏳{{DAYS_LEFT}}D")
  204. first := s.genHostRemark(inbound, client, "", "")
  205. second := s.genHostRemark(inbound, client, "", "")
  206. if !strings.Contains(first, "📊") || !strings.Contains(first, "80.00GB") {
  207. t.Fatalf("first link should carry usage: %q", first)
  208. }
  209. if strings.ContainsAny(second, "📊⏳") {
  210. t.Fatalf("second link must not carry usage: %q", second)
  211. }
  212. if second != "DE" {
  213. t.Fatalf("second link = %q, want name-only %q", second, "DE")
  214. }
  215. }
  216. // Outside the subscription body (panel link/QR displays, sub info page) the
  217. // template is bypassed entirely — links show just the config name, with no
  218. // per-client email or usage info.
  219. func TestRemarkInDisplayContext(t *testing.T) {
  220. s, inbound, client := hostRemarkService("{{INBOUND}}|📊{{TRAFFIC_LEFT}}|⏳{{DAYS_LEFT}}D")
  221. s.subscriptionBody = false
  222. // A host link in a display shows only the config name — the inbound's remark,
  223. // with no per-client email or usage info and the host remark ignored.
  224. if got := s.genHostRemark(inbound, client, "CDN", ""); got != "DE" {
  225. t.Fatalf("display host link = %q, want config name %q", got, "DE")
  226. }
  227. // With no host remark, the config name is likewise the inbound's own remark.
  228. if got := s.genHostRemark(inbound, client, "", ""); got != "DE" {
  229. t.Fatalf("display host link (no host) = %q, want %q", got, "DE")
  230. }
  231. // genRemark (non-host) likewise drops the template in display context.
  232. if got := s.genRemark(inbound, client.Email, "", ""); got != "DE" {
  233. t.Fatalf("display genRemark = %q, want %q", got, "DE")
  234. }
  235. }
  236. // nameOnlyTemplate drops the info part (and its leading decoration), keeping name.
  237. func TestNameOnlyTemplate(t *testing.T) {
  238. cases := map[string]string{
  239. "{{INBOUND}}|📊{{TRAFFIC_LEFT}}|⏳{{DAYS_LEFT}}D": "{{INBOUND}}", // the default → name only
  240. "{{EMAIL}} {{INBOUND}} ⏳{{DAYS_LEFT}}": "{{EMAIL}} {{INBOUND}}", // multi-token name survives the trim
  241. "{{INBOUND}} | {{STATUS}}": "{{INBOUND}}",
  242. "{{INBOUND}}-{{EMAIL}}": "{{INBOUND}}-{{EMAIL}}", // no info tokens → unchanged
  243. "{{TRAFFIC_LEFT}}": "", // info only → empty
  244. }
  245. for tmpl, want := range cases {
  246. if got := nameOnlyTemplate(tmpl); got != want {
  247. t.Errorf("nameOnlyTemplate(%q) = %q, want %q", tmpl, got, want)
  248. }
  249. }
  250. }
  251. // statsForClient resolves usage from the per-request statsByEmail map when the
  252. // link's own inbound doesn't carry the client's (globally unique) traffic row —
  253. // the multi-inbound case that made {{TRAFFIC_LEFT}} show the full quota (#5443).
  254. func TestStatsForClient_CrossInboundFallback(t *testing.T) {
  255. s := &SubService{
  256. statsByEmail: map[string]xray.ClientTraffic{
  257. "[email protected]": {Email: "[email protected]", Total: 100 * gb, Up: 15 * gb, Down: 5 * gb},
  258. },
  259. }
  260. // Inbound B carries no ClientStats for john (his row is owned by inbound A).
  261. inboundB := &model.Inbound{Remark: "B"}
  262. st := s.statsForClient(inboundB, model.Client{Email: "[email protected]"})
  263. if used := st.Up + st.Down; used != 20*gb {
  264. t.Fatalf("statsForClient used = %d, want %d (cross-inbound fallback)", used, 20*gb)
  265. }
  266. if got := remarkVarValue("TRAFFIC_LEFT", remarkContext{stats: st}); got != "80.00GB" {
  267. t.Fatalf("TRAFFIC_LEFT = %q, want 80.00GB (remaining, not total)", got)
  268. }
  269. }
  270. // Two clients through the same global template get distinct, per-client remarks.
  271. func TestGenHostRemark_PerClient(t *testing.T) {
  272. s := &SubService{remarkTemplate: "{{EMAIL}}", subscriptionBody: true}
  273. inbound := &model.Inbound{}
  274. a := s.genHostRemark(inbound, model.Client{Email: "alice@x"}, "", "")
  275. b := s.genHostRemark(inbound, model.Client{Email: "bob@x"}, "", "")
  276. if a != "alice@x" || b != "bob@x" {
  277. t.Fatalf("per-client expansion failed: a=%q b=%q", a, b)
  278. }
  279. }
  280. func TestStatusEmoji(t *testing.T) {
  281. cases := []struct {
  282. stats xray.ClientTraffic
  283. want string
  284. }{
  285. {xray.ClientTraffic{Enable: true, Total: 10 * gb, Up: gb}, "✅"},
  286. {xray.ClientTraffic{Enable: true, Total: 10 * gb, Up: 10 * gb, Down: 1}, "🚫"},
  287. {xray.ClientTraffic{Enable: false}, "🚫"},
  288. {xray.ClientTraffic{Enable: true, ExpiryTime: 1000}, "⏳"},
  289. }
  290. for _, c := range cases {
  291. if got := statusEmoji(c.stats); got != c.want {
  292. t.Errorf("statusEmoji(%+v) = %q, want %q", c.stats, got, c.want)
  293. }
  294. }
  295. }
  296. func TestUsagePercentage(t *testing.T) {
  297. if got := usagePercentage(xray.ClientTraffic{Total: 100 * gb, Up: 25 * gb, Down: 25 * gb}); got != "50.0%" {
  298. t.Errorf("usagePercentage 50%% = %q", got)
  299. }
  300. if got := usagePercentage(xray.ClientTraffic{Total: 0}); got != "" {
  301. t.Errorf("usagePercentage unlimited = %q, want empty", got)
  302. }
  303. if got := usagePercentage(xray.ClientTraffic{Total: 10 * gb, Up: 10 * gb}); got != "100.0%" {
  304. t.Errorf("usagePercentage 100%% = %q", got)
  305. }
  306. // Over-quota usage clamps to 100%, consistent with TRAFFIC_LEFT.
  307. if got := usagePercentage(xray.ClientTraffic{Total: 10 * gb, Up: 25 * gb}); got != "100.0%" {
  308. t.Errorf("usagePercentage over-quota = %q, want 100.0%%", got)
  309. }
  310. }
  311. func TestTimeLeftLabel(t *testing.T) {
  312. if got := timeLeftLabel(0); got != "∞" {
  313. t.Errorf("timeLeftLabel(0) = %q, want ∞", got)
  314. }
  315. // Delayed-start: negative expiry = duration in ms. 1000ms = 1 second = "0m".
  316. if got := timeLeftLabel(-1000); got != "0m" {
  317. t.Errorf("timeLeftLabel(-1000) = %q, want 0m", got)
  318. }
  319. }
  320. func TestGregorianToJalali(t *testing.T) {
  321. cases := []struct {
  322. gy, gm, gd int
  323. jy, jm, jd int
  324. }{
  325. {2024, 1, 1, 1402, 10, 11},
  326. {2000, 3, 20, 1379, 1, 1},
  327. {1979, 2, 11, 1357, 11, 22},
  328. }
  329. for _, c := range cases {
  330. jy, jm, jd := gregorianToJalali(c.gy, c.gm, c.gd)
  331. if jy != c.jy || jm != c.jm || jd != c.jd {
  332. t.Errorf("gregorianToJalali(%d,%d,%d) = (%d,%d,%d), want (%d,%d,%d)",
  333. c.gy, c.gm, c.gd, jy, jm, jd, c.jy, c.jm, c.jd)
  334. }
  335. }
  336. }
  337. func TestJalaliExpireDateLabel(t *testing.T) {
  338. if got := jalaliExpireDateLabel(0); got != "" {
  339. t.Errorf("jalaliExpireDateLabel(0) = %q, want empty", got)
  340. }
  341. if got := jalaliExpireDateLabel(-1000); got != "" {
  342. t.Errorf("jalaliExpireDateLabel(-1000) = %q, want empty", got)
  343. }
  344. }
  345. func TestExpandNewTokensInTemplate(t *testing.T) {
  346. inbound := &model.Inbound{Remark: "DE", Protocol: "vless"}
  347. client := model.Client{Email: "[email protected]", ID: "abc-123"}
  348. stats := xray.ClientTraffic{Enable: true, Total: 100 * gb, Up: 50 * gb, Down: 0}
  349. ctx := remarkContext{
  350. client: client,
  351. stats: stats,
  352. inbound: inbound,
  353. transport: "ws",
  354. }
  355. cases := []struct{ tmpl, want string }{
  356. {"{{STATUS_EMOJI}}", "✅"},
  357. {"{{USAGE_PERCENTAGE}}", "50.0%"},
  358. {"{{PROTOCOL}}", "VLESS"},
  359. {"{{TRANSPORT}}", "ws"},
  360. {"{{STATUS_EMOJI}} {{INBOUND}}", "✅ DE"},
  361. }
  362. for _, c := range cases {
  363. if got := expandRemarkVars(c.tmpl, ctx); got != c.want {
  364. t.Errorf("expandRemarkVars(%q) = %q, want %q", c.tmpl, got, c.want)
  365. }
  366. }
  367. }
  368. func TestTranslateUISingleBrackets(t *testing.T) {
  369. cases := []struct{ in, want string }{
  370. {"{EMAIL}", "{{EMAIL}}"},
  371. {"{DATA_LEFT}", "{{TRAFFIC_LEFT}}"},
  372. {"{DATA_LEFT} of {DATA_LIMIT}", "{{TRAFFIC_LEFT}} of {{TRAFFIC_TOTAL}}"},
  373. {"{STATUS_EMOJI} {INBOUND}", "{{STATUS_EMOJI}} {INBOUND}"},
  374. {"{UNKNOWN_TOKEN}", "{UNKNOWN_TOKEN}"},
  375. {"no braces", "no braces"},
  376. {"{{TRAFFIC_LEFT}}", "{{TRAFFIC_LEFT}}"},
  377. {"{username}", "{username}"},
  378. }
  379. for _, c := range cases {
  380. if got := translateUISingleBrackets(c.in); got != c.want {
  381. t.Errorf("translateUISingleBrackets(%q) = %q, want %q", c.in, got, c.want)
  382. }
  383. }
  384. }
  385. func TestExpandRemarkVars_SingleBracketUI(t *testing.T) {
  386. inbound := &model.Inbound{Remark: "DE", Protocol: "vless"}
  387. stats := xray.ClientTraffic{Enable: true, Total: 100 * gb, Up: 50 * gb, Down: 0}
  388. ctx := remarkContext{
  389. client: model.Client{Email: "[email protected]"},
  390. stats: stats,
  391. inbound: inbound,
  392. transport: "ws",
  393. }
  394. cases := []struct{ tmpl, want string }{
  395. {"{EMAIL}", "[email protected]"},
  396. {"{DATA_LEFT}", "50.00GB"},
  397. {"{DATA_USAGE}", "50.00GB"},
  398. {"{DATA_LIMIT}", "100.00GB"},
  399. {"{STATUS_EMOJI}", "✅"},
  400. {"{USAGE_PERCENTAGE}", "50.0%"},
  401. {"{PROTOCOL}", "VLESS"},
  402. {"{TRANSPORT}", "ws"},
  403. }
  404. for _, c := range cases {
  405. if got := expandRemarkVars(c.tmpl, ctx); got != c.want {
  406. t.Errorf("expandRemarkVars(%q) = %q, want %q", c.tmpl, got, c.want)
  407. }
  408. }
  409. }
  410. func TestUsageOnFirstLinkOnly_SingleBracket(t *testing.T) {
  411. s := &SubService{
  412. remarkTemplate: "{STATUS_EMOJI} {{INBOUND}}|📊{{TRAFFIC_LEFT}}",
  413. subscriptionBody: true,
  414. usageShown: map[string]bool{},
  415. }
  416. inbound := &model.Inbound{
  417. Remark: "DE",
  418. ClientStats: []xray.ClientTraffic{{
  419. Email: "alice@x",
  420. Enable: true,
  421. Total: 100 * gb,
  422. Up: 20 * gb,
  423. Down: 10 * gb,
  424. }},
  425. }
  426. client := model.Client{Email: "alice@x"}
  427. first := s.genTemplatedRemark(inbound, client, "", "ws")
  428. s.usageShown["alice@x"] = true
  429. second := s.genTemplatedRemark(inbound, client, "", "ws")
  430. if !strings.Contains(first, "📊") {
  431. t.Fatalf("first link should carry usage: %q", first)
  432. }
  433. if strings.Contains(second, "📊") {
  434. t.Fatalf("second link must not carry usage: %q", second)
  435. }
  436. }