package sing import ( "encoding/base64" "errors" "github.com/InazumaV/V2bX/api/panel" "github.com/InazumaV/V2bX/common/counter" "github.com/InazumaV/V2bX/core" "github.com/sagernet/sing-box/option" "github.com/sagernet/sing-box/protocol/hysteria" "github.com/sagernet/sing-box/protocol/hysteria2" "github.com/sagernet/sing-box/protocol/shadowsocks" "github.com/sagernet/sing-box/protocol/trojan" "github.com/sagernet/sing-box/protocol/vless" "github.com/sagernet/sing-box/protocol/vmess" ) func (b *Sing) AddUsers(p *core.AddUsersParams) (added int, err error) { in, found := b.box.Inbound().Get(p.Tag) if !found { return 0, errors.New("the inbound not found") } switch p.NodeInfo.Type { case "vless": us := make([]option.VLESSUser, len(p.Users)) for i := range p.Users { us[i] = option.VLESSUser{ Name: p.Users[i].Uuid, Flow: p.VAllss.Flow, UUID: p.Users[i].Uuid, } } err = in.(*vless.Inbound).AddUsers(us) case "vmess": us := make([]option.VMessUser, len(p.Users)) for i := range p.Users { us[i] = option.VMessUser{ Name: p.Users[i].Uuid, UUID: p.Users[i].Uuid, } } err = in.(*vmess.Inbound).AddUsers(us) case "shadowsocks": us := make([]option.ShadowsocksUser, len(p.Users)) for i := range p.Users { var password = p.Users[i].Uuid switch p.Shadowsocks.Cipher { case "2022-blake3-aes-128-gcm": password = base64.StdEncoding.EncodeToString([]byte(password[:16])) case "2022-blake3-aes-256-gcm": password = base64.StdEncoding.EncodeToString([]byte(password[:32])) } us[i] = option.ShadowsocksUser{ Name: p.Users[i].Uuid, Password: password, } } err = in.(*shadowsocks.MultiInbound).AddUsers(us) case "trojan": us := make([]option.TrojanUser, len(p.Users)) for i := range p.Users { us[i] = option.TrojanUser{ Name: p.Users[i].Uuid, Password: p.Users[i].Uuid, } } err = in.(*trojan.Inbound).AddUsers(us) case "hysteria": us := make([]option.HysteriaUser, len(p.Users)) for i := range p.Users { us[i] = option.HysteriaUser{ Name: p.Users[i].Uuid, AuthString: p.Users[i].Uuid, } } err = in.(*hysteria.Inbound).AddUsers(us) case "hysteria2": us := make([]option.Hysteria2User, len(p.Users)) id := make([]int, len(p.Users)) for i := range p.Users { us[i] = option.Hysteria2User{ Name: p.Users[i].Uuid, Password: p.Users[i].Uuid, } id[i] = p.Users[i].Id } err = in.(*hysteria2.Inbound).AddUsers(us, id) } if err != nil { return 0, err } return len(p.Users), err } func (b *Sing) GetUserTraffic(tag, uuid string, reset bool) (up int64, down int64) { if v, ok := b.hookServer.counter.Load(tag); ok { c := v.(*counter.TrafficCounter) up = c.GetUpCount(uuid) down = c.GetDownCount(uuid) if reset { c.Reset(uuid) } return } return 0, 0 } type UserDeleter interface { DelUsers(uuid []string) error } func (b *Sing) DelUsers(users []panel.UserInfo, tag string, info *panel.NodeInfo) error { var del UserDeleter if i, found := b.box.Inbound().Get(tag); found { switch info.Type { case "vmess": del = i.(*vmess.Inbound) case "vless": del = i.(*vless.Inbound) case "shadowsocks": del = i.(*shadowsocks.MultiInbound) case "trojan": del = i.(*trojan.Inbound) case "hysteria": del = i.(*hysteria.Inbound) case "hysteria2": del = i.(*hysteria2.Inbound) } } else { return errors.New("the inbound not found") } uuids := make([]string, len(users)) for i := range users { uuids[i] = users[i].Uuid } err := del.DelUsers(uuids) if err != nil { return err } return nil }