Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
51c3807d01 | ||
|
|
c2c42ca2b7 | ||
|
|
2a6db6ebfe | ||
|
|
30d8edbdcf | ||
|
|
177afafe12 | ||
|
|
98765b6e2a |
@@ -2,7 +2,7 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"flag"
|
"flag"
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/utils"
|
"github.com/TeaOSLab/EdgeAPI/internal/installers/helpers"
|
||||||
"github.com/iwind/gosock/pkg/gosock"
|
"github.com/iwind/gosock/pkg/gosock"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
@@ -51,7 +51,7 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
unzip := utils.NewUnzip(zipPath, targetPath)
|
unzip := helpers.NewUnzip(zipPath, targetPath)
|
||||||
err := unzip.Run()
|
err := unzip.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
stderr("ERROR: " + err.Error())
|
stderr("ERROR: " + err.Error())
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
package main
|
package main
|
||||||
|
|
||||||
|
// 注意这里的依赖文件应该最小化,从而使编译后的文件最小化
|
||||||
import (
|
import (
|
||||||
"flag"
|
"flag"
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/utils"
|
"github.com/TeaOSLab/EdgeAPI/internal/installers/helpers"
|
||||||
"github.com/iwind/gosock/pkg/gosock"
|
"github.com/iwind/gosock/pkg/gosock"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
@@ -51,7 +52,7 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
unzip := utils.NewUnzip(zipPath, targetPath)
|
unzip := helpers.NewUnzip(zipPath, targetPath)
|
||||||
err := unzip.Run()
|
err := unzip.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
stderr("ERROR: " + err.Error())
|
stderr("ERROR: " + err.Error())
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
package teaconst
|
package teaconst
|
||||||
|
|
||||||
const (
|
const (
|
||||||
Version = "0.5.4"
|
Version = "0.5.5"
|
||||||
|
|
||||||
ProductName = "Edge API"
|
ProductName = "Edge API"
|
||||||
ProcessName = "edge-api"
|
ProcessName = "edge-api"
|
||||||
@@ -18,8 +18,8 @@ const (
|
|||||||
|
|
||||||
// 其他节点版本号,用来检测是否有需要升级的节点
|
// 其他节点版本号,用来检测是否有需要升级的节点
|
||||||
|
|
||||||
NodeVersion = "0.5.4"
|
NodeVersion = "0.5.5"
|
||||||
UserNodeVersion = "0.5.0"
|
UserNodeVersion = "0.5.5"
|
||||||
DNSNodeVersion = "0.2.8"
|
DNSNodeVersion = "0.2.8"
|
||||||
AuthorityNodeVersion = "0.0.2"
|
AuthorityNodeVersion = "0.0.2"
|
||||||
MonitorNodeVersion = "0.0.4"
|
MonitorNodeVersion = "0.0.4"
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ type OrderMethod struct {
|
|||||||
Secret string `field:"secret"` // 密钥
|
Secret string `field:"secret"` // 密钥
|
||||||
Params dbs.JSON `field:"params"` // 参数
|
Params dbs.JSON `field:"params"` // 参数
|
||||||
ClientType string `field:"clientType"` // 客户端类型
|
ClientType string `field:"clientType"` // 客户端类型
|
||||||
|
QrcodeTitle string `field:"qrcodeTitle"` // 二维码标题
|
||||||
Order uint32 `field:"order"` // 排序
|
Order uint32 `field:"order"` // 排序
|
||||||
State uint8 `field:"state"` // 状态
|
State uint8 `field:"state"` // 状态
|
||||||
}
|
}
|
||||||
@@ -29,6 +30,7 @@ type OrderMethodOperator struct {
|
|||||||
Secret any // 密钥
|
Secret any // 密钥
|
||||||
Params any // 参数
|
Params any // 参数
|
||||||
ClientType any // 客户端类型
|
ClientType any // 客户端类型
|
||||||
|
QrcodeTitle any // 二维码标题
|
||||||
Order any // 排序
|
Order any // 排序
|
||||||
State any // 状态
|
State any // 状态
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,80 +0,0 @@
|
|||||||
package accounts
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/userconfigs"
|
|
||||||
_ "github.com/go-sql-driver/mysql"
|
|
||||||
"github.com/iwind/TeaGo/Tea"
|
|
||||||
"github.com/iwind/TeaGo/dbs"
|
|
||||||
"github.com/iwind/TeaGo/maps"
|
|
||||||
timeutil "github.com/iwind/TeaGo/utils/time"
|
|
||||||
)
|
|
||||||
|
|
||||||
type UserAccountDailyStatDAO dbs.DAO
|
|
||||||
|
|
||||||
func NewUserAccountDailyStatDAO() *UserAccountDailyStatDAO {
|
|
||||||
return dbs.NewDAO(&UserAccountDailyStatDAO{
|
|
||||||
DAOObject: dbs.DAOObject{
|
|
||||||
DB: Tea.Env,
|
|
||||||
Table: "edgeUserAccountDailyStats",
|
|
||||||
Model: new(UserAccountDailyStat),
|
|
||||||
PkName: "id",
|
|
||||||
},
|
|
||||||
}).(*UserAccountDailyStatDAO)
|
|
||||||
}
|
|
||||||
|
|
||||||
var SharedUserAccountDailyStatDAO *UserAccountDailyStatDAO
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
dbs.OnReady(func() {
|
|
||||||
SharedUserAccountDailyStatDAO = NewUserAccountDailyStatDAO()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdateDailyStat 更新当天统计数据
|
|
||||||
func (this *UserAccountDailyStatDAO) UpdateDailyStat(tx *dbs.Tx) error {
|
|
||||||
var day = timeutil.Format("Ymd")
|
|
||||||
var month = timeutil.Format("Ym")
|
|
||||||
income, err := SharedUserAccountLogDAO.SumDailyEventTypes(tx, day, userconfigs.AccountIncomeEventTypes)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
expense, err := SharedUserAccountLogDAO.SumDailyEventTypes(tx, day, userconfigs.AccountExpenseEventTypes)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if expense < 0 {
|
|
||||||
expense = -expense
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.Query(tx).
|
|
||||||
InsertOrUpdateQuickly(maps.Map{
|
|
||||||
"day": day,
|
|
||||||
"month": month,
|
|
||||||
"income": income,
|
|
||||||
"expense": expense,
|
|
||||||
}, maps.Map{
|
|
||||||
"income": income,
|
|
||||||
"expense": expense,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// FindDailyStats 查看按天统计
|
|
||||||
func (this *UserAccountDailyStatDAO) FindDailyStats(tx *dbs.Tx, dayFrom string, dayTo string) (result []*UserAccountDailyStat, err error) {
|
|
||||||
_, err = this.Query(tx).
|
|
||||||
Between("day", dayFrom, dayTo).
|
|
||||||
Slice(&result).
|
|
||||||
FindAll()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// FindMonthlyStats 查看某月统计
|
|
||||||
func (this *UserAccountDailyStatDAO) FindMonthlyStats(tx *dbs.Tx, dayFrom string, dayTo string) (result []*UserAccountDailyStat, err error) {
|
|
||||||
_, err = this.Query(tx).
|
|
||||||
Result("SUM(income) AS income", "SUM(expense) AS expense", "month").
|
|
||||||
Between("day", dayFrom, dayTo).
|
|
||||||
Group("month").
|
|
||||||
Slice(&result).
|
|
||||||
FindAll()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
package accounts
|
|
||||||
|
|
||||||
import (
|
|
||||||
_ "github.com/go-sql-driver/mysql"
|
|
||||||
_ "github.com/iwind/TeaGo/bootstrap"
|
|
||||||
)
|
|
||||||
@@ -1,253 +0,0 @@
|
|||||||
package accounts
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/db/models"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/errors"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/goman"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/remotelogs"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/userconfigs"
|
|
||||||
_ "github.com/go-sql-driver/mysql"
|
|
||||||
"github.com/iwind/TeaGo/Tea"
|
|
||||||
"github.com/iwind/TeaGo/dbs"
|
|
||||||
"github.com/iwind/TeaGo/lists"
|
|
||||||
"github.com/iwind/TeaGo/maps"
|
|
||||||
"github.com/iwind/TeaGo/types"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
dbs.OnReadyDone(func() {
|
|
||||||
goman.New(func() {
|
|
||||||
// 自动支付账单任务
|
|
||||||
var ticker = time.NewTicker(12 * time.Hour)
|
|
||||||
for range ticker.C {
|
|
||||||
if SharedUserAccountDAO.Instance != nil {
|
|
||||||
err := SharedUserAccountDAO.Instance.RunTx(func(tx *dbs.Tx) error {
|
|
||||||
return SharedUserAccountDAO.PayBills(tx)
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
remotelogs.Error("USER_ACCOUNT_DAO", "pay bills task failed: "+err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
type UserAccountDAO dbs.DAO
|
|
||||||
|
|
||||||
func NewUserAccountDAO() *UserAccountDAO {
|
|
||||||
return dbs.NewDAO(&UserAccountDAO{
|
|
||||||
DAOObject: dbs.DAOObject{
|
|
||||||
DB: Tea.Env,
|
|
||||||
Table: "edgeUserAccounts",
|
|
||||||
Model: new(UserAccount),
|
|
||||||
PkName: "id",
|
|
||||||
},
|
|
||||||
}).(*UserAccountDAO)
|
|
||||||
}
|
|
||||||
|
|
||||||
var SharedUserAccountDAO *UserAccountDAO
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
dbs.OnReady(func() {
|
|
||||||
SharedUserAccountDAO = NewUserAccountDAO()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// FindUserAccountWithUserId 根据用户ID查找用户账户
|
|
||||||
func (this *UserAccountDAO) FindUserAccountWithUserId(tx *dbs.Tx, userId int64) (*UserAccount, error) {
|
|
||||||
if userId <= 0 {
|
|
||||||
return nil, errors.New("invalid userId '" + types.String(userId) + "'")
|
|
||||||
}
|
|
||||||
|
|
||||||
// 用户是否存在
|
|
||||||
user, err := models.SharedUserDAO.FindEnabledUser(tx, userId, nil)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if user == nil {
|
|
||||||
return nil, errors.New("invalid userId '" + types.String(userId) + "'")
|
|
||||||
}
|
|
||||||
|
|
||||||
account, err := this.Query(tx).
|
|
||||||
Attr("userId", userId).
|
|
||||||
Find()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if account != nil {
|
|
||||||
return account.(*UserAccount), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
var op = NewUserAccountOperator()
|
|
||||||
op.UserId = userId
|
|
||||||
_, err = this.SaveInt64(tx, op)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return this.FindUserAccountWithUserId(tx, userId)
|
|
||||||
}
|
|
||||||
|
|
||||||
// FindUserAccountWithAccountId 根据ID查找用户账户
|
|
||||||
func (this *UserAccountDAO) FindUserAccountWithAccountId(tx *dbs.Tx, accountId int64) (*UserAccount, error) {
|
|
||||||
one, err := this.Query(tx).
|
|
||||||
Pk(accountId).
|
|
||||||
Find()
|
|
||||||
if one != nil {
|
|
||||||
return one.(*UserAccount), nil
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdateUserAccount 操作用户账户
|
|
||||||
func (this *UserAccountDAO) UpdateUserAccount(tx *dbs.Tx, accountId int64, delta float32, eventType userconfigs.AccountEventType, description string, params maps.Map) error {
|
|
||||||
account, err := this.FindUserAccountWithAccountId(tx, accountId)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if account == nil {
|
|
||||||
return errors.New("invalid account id '" + types.String(accountId) + "'")
|
|
||||||
}
|
|
||||||
var userId = int64(account.UserId)
|
|
||||||
var deltaFloat64 = float64(delta)
|
|
||||||
if deltaFloat64 < 0 && account.Total < -deltaFloat64 {
|
|
||||||
return errors.New("not enough account quota to decrease")
|
|
||||||
}
|
|
||||||
|
|
||||||
// 操作账户
|
|
||||||
err = this.Query(tx).
|
|
||||||
Pk(account.Id).
|
|
||||||
Set("total", dbs.SQL("total+:delta")).
|
|
||||||
Param("delta", delta).
|
|
||||||
UpdateQuickly()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// 生成日志
|
|
||||||
err = SharedUserAccountLogDAO.CreateAccountLog(tx, userId, accountId, delta, 0, eventType, description, params)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdateUserAccountFrozen 操作用户账户冻结余额
|
|
||||||
func (this *UserAccountDAO) UpdateUserAccountFrozen(tx *dbs.Tx, userId int64, delta float32, eventType userconfigs.AccountEventType, description string, params maps.Map) error {
|
|
||||||
account, err := this.FindUserAccountWithUserId(tx, userId)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
var deltaFloat64 = float64(delta)
|
|
||||||
if deltaFloat64 < 0 && account.TotalFrozen < -deltaFloat64 {
|
|
||||||
return errors.New("not enough account frozen quota to decrease")
|
|
||||||
}
|
|
||||||
|
|
||||||
// 操作账户
|
|
||||||
err = this.Query(tx).
|
|
||||||
Pk(account.Id).
|
|
||||||
Set("totalFrozen", dbs.SQL("total+:delta")).
|
|
||||||
Param("delta", delta).
|
|
||||||
UpdateQuickly()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// 生成日志
|
|
||||||
err = SharedUserAccountLogDAO.CreateAccountLog(tx, userId, int64(account.Id), 0, delta, eventType, description, params)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// CountAllAccounts 计算所有账户数量
|
|
||||||
func (this *UserAccountDAO) CountAllAccounts(tx *dbs.Tx, keyword string) (int64, error) {
|
|
||||||
var query = this.Query(tx)
|
|
||||||
if len(keyword) > 0 {
|
|
||||||
query.Where("userId IN (SELECT id FROM " + models.SharedUserDAO.Table + " WHERE state=1 AND (username LIKE :keyword OR fullname LIKE :keyword))")
|
|
||||||
query.Param("keyword", keyword)
|
|
||||||
} else {
|
|
||||||
query.Where("userId IN (SELECT id FROM " + models.SharedUserDAO.Table + " WHERE state=1)")
|
|
||||||
}
|
|
||||||
return query.Count()
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListAccounts 列出单页账户
|
|
||||||
func (this *UserAccountDAO) ListAccounts(tx *dbs.Tx, keyword string, offset int64, size int64) (result []*UserAccount, err error) {
|
|
||||||
var query = this.Query(tx)
|
|
||||||
if len(keyword) > 0 {
|
|
||||||
query.Where("userId IN (SELECT id FROM " + models.SharedUserDAO.Table + " WHERE state=1 AND (username LIKE :keyword OR fullname LIKE :keyword))")
|
|
||||||
query.Param("keyword", keyword)
|
|
||||||
} else {
|
|
||||||
query.Where("userId IN (SELECT id FROM " + models.SharedUserDAO.Table + " WHERE state=1)")
|
|
||||||
}
|
|
||||||
_, err = query.
|
|
||||||
DescPk().
|
|
||||||
Offset(offset).
|
|
||||||
Limit(size).
|
|
||||||
Slice(&result).
|
|
||||||
FindAll()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// PayBills 尝试自动支付账单
|
|
||||||
func (this *UserAccountDAO) PayBills(tx *dbs.Tx) error {
|
|
||||||
bills, err := models.SharedUserBillDAO.FindUnpaidBills(tx, 10000)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// 先支付久远的
|
|
||||||
lists.Reverse(bills)
|
|
||||||
|
|
||||||
for _, bill := range bills {
|
|
||||||
if bill.Amount <= 0 {
|
|
||||||
err = models.SharedUserBillDAO.UpdateUserBillIsPaid(tx, int64(bill.Id), true)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
account, err := SharedUserAccountDAO.FindUserAccountWithUserId(tx, int64(bill.UserId))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if account == nil || account.Total < bill.Amount {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
// 扣款
|
|
||||||
err = SharedUserAccountDAO.UpdateUserAccount(tx, int64(account.Id), -float32(bill.Amount), userconfigs.AccountEventTypePayBill, "支付账单"+bill.Code, maps.Map{"billId": bill.Id})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// 改为已支付
|
|
||||||
err = models.SharedUserBillDAO.UpdateUserBillIsPaid(tx, int64(bill.Id), true)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// CheckUserAccount 检查用户账户
|
|
||||||
func (this *UserAccountDAO) CheckUserAccount(tx *dbs.Tx, userId int64, accountId int64) error {
|
|
||||||
exists, err := this.Query(tx).
|
|
||||||
Pk(accountId).
|
|
||||||
Attr("userId", userId).
|
|
||||||
Exist()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if !exists {
|
|
||||||
return models.ErrNotFound
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
package accounts
|
|
||||||
|
|
||||||
import (
|
|
||||||
_ "github.com/go-sql-driver/mysql"
|
|
||||||
_ "github.com/iwind/TeaGo/bootstrap"
|
|
||||||
"github.com/iwind/TeaGo/dbs"
|
|
||||||
"testing"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestUserAccountDAO_PayBills(t *testing.T) {
|
|
||||||
dbs.NotifyReady()
|
|
||||||
|
|
||||||
err := NewUserAccountDAO().PayBills(nil)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
t.Log("ok")
|
|
||||||
}
|
|
||||||
@@ -1,129 +0,0 @@
|
|||||||
package accounts
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/db/models"
|
|
||||||
dbutils "github.com/TeaOSLab/EdgeAPI/internal/db/utils"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/errors"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/userconfigs"
|
|
||||||
_ "github.com/go-sql-driver/mysql"
|
|
||||||
"github.com/iwind/TeaGo/Tea"
|
|
||||||
"github.com/iwind/TeaGo/dbs"
|
|
||||||
"github.com/iwind/TeaGo/maps"
|
|
||||||
"github.com/iwind/TeaGo/types"
|
|
||||||
timeutil "github.com/iwind/TeaGo/utils/time"
|
|
||||||
)
|
|
||||||
|
|
||||||
type UserAccountLogDAO dbs.DAO
|
|
||||||
|
|
||||||
func NewUserAccountLogDAO() *UserAccountLogDAO {
|
|
||||||
return dbs.NewDAO(&UserAccountLogDAO{
|
|
||||||
DAOObject: dbs.DAOObject{
|
|
||||||
DB: Tea.Env,
|
|
||||||
Table: "edgeUserAccountLogs",
|
|
||||||
Model: new(UserAccountLog),
|
|
||||||
PkName: "id",
|
|
||||||
},
|
|
||||||
}).(*UserAccountLogDAO)
|
|
||||||
}
|
|
||||||
|
|
||||||
var SharedUserAccountLogDAO *UserAccountLogDAO
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
dbs.OnReady(func() {
|
|
||||||
SharedUserAccountLogDAO = NewUserAccountLogDAO()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// CreateAccountLog 生成用户账户日志
|
|
||||||
func (this *UserAccountLogDAO) CreateAccountLog(tx *dbs.Tx, userId int64, accountId int64, delta float32, deltaFrozen float32, eventType userconfigs.AccountEventType, description string, params maps.Map) error {
|
|
||||||
var op = NewUserAccountLogOperator()
|
|
||||||
op.UserId = userId
|
|
||||||
op.AccountId = accountId
|
|
||||||
op.Delta = delta
|
|
||||||
op.DeltaFrozen = deltaFrozen
|
|
||||||
|
|
||||||
account, err := SharedUserAccountDAO.FindUserAccountWithAccountId(tx, accountId)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if account == nil {
|
|
||||||
return errors.New("invalid account id '" + types.String(accountId) + "'")
|
|
||||||
}
|
|
||||||
op.Total = account.Total
|
|
||||||
op.TotalFrozen = account.TotalFrozen
|
|
||||||
|
|
||||||
op.EventType = eventType
|
|
||||||
op.Description = description
|
|
||||||
|
|
||||||
if params == nil {
|
|
||||||
params = maps.Map{}
|
|
||||||
}
|
|
||||||
op.Params = params.AsJSON()
|
|
||||||
|
|
||||||
op.Day = timeutil.Format("Ymd")
|
|
||||||
err = this.Save(tx, op)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
return SharedUserAccountDailyStatDAO.UpdateDailyStat(tx)
|
|
||||||
}
|
|
||||||
|
|
||||||
// CountAccountLogs 计算日志数量
|
|
||||||
func (this *UserAccountLogDAO) CountAccountLogs(tx *dbs.Tx, userId int64, accountId int64, keyword string, eventType string) (int64, error) {
|
|
||||||
var query = this.Query(tx)
|
|
||||||
if userId > 0 {
|
|
||||||
query.Attr("userId", userId)
|
|
||||||
}
|
|
||||||
if accountId > 0 {
|
|
||||||
query.Attr("accountId", accountId)
|
|
||||||
}
|
|
||||||
if len(keyword) > 0 {
|
|
||||||
query.Where("(userId IN (SELECT id FROM " + models.SharedUserDAO.Table + " WHERE state=1 AND (username LIKE :keyword OR fullname LIKE :keyword)) OR description LIKE :keyword)")
|
|
||||||
query.Param("keyword", dbutils.QuoteLike(keyword))
|
|
||||||
}
|
|
||||||
if len(eventType) > 0 {
|
|
||||||
query.Attr("eventType", eventType)
|
|
||||||
}
|
|
||||||
return query.Count()
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListAccountLogs 列出单页日志
|
|
||||||
func (this *UserAccountLogDAO) ListAccountLogs(tx *dbs.Tx, userId int64, accountId int64, keyword string, eventType string, offset int64, size int64) (result []*UserAccountLog, err error) {
|
|
||||||
var query = this.Query(tx)
|
|
||||||
if userId > 0 {
|
|
||||||
query.Attr("userId", userId)
|
|
||||||
}
|
|
||||||
if accountId > 0 {
|
|
||||||
query.Attr("accountId", accountId)
|
|
||||||
}
|
|
||||||
if len(keyword) > 0 {
|
|
||||||
query.Where("(userId IN (SELECT id FROM " + models.SharedUserDAO.Table + " WHERE state=1 AND (username LIKE :keyword OR fullname LIKE :keyword)) OR description LIKE :keyword)")
|
|
||||||
query.Param("keyword", dbutils.QuoteLike(keyword))
|
|
||||||
}
|
|
||||||
if len(eventType) > 0 {
|
|
||||||
query.Attr("eventType", eventType)
|
|
||||||
}
|
|
||||||
_, err = query.
|
|
||||||
DescPk().
|
|
||||||
Offset(offset).
|
|
||||||
Limit(size).
|
|
||||||
Slice(&result).
|
|
||||||
FindAll()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// SumDailyEventTypes 统计某天数据总和
|
|
||||||
func (this *UserAccountLogDAO) SumDailyEventTypes(tx *dbs.Tx, day string, eventTypes []userconfigs.AccountEventType) (float32, error) {
|
|
||||||
if len(eventTypes) == 0 {
|
|
||||||
return 0, nil
|
|
||||||
}
|
|
||||||
result, err := this.Query(tx).
|
|
||||||
Attr("day", day).
|
|
||||||
Attr("eventType", eventTypes).
|
|
||||||
Sum("delta", 0)
|
|
||||||
if err != nil {
|
|
||||||
return 0, err
|
|
||||||
}
|
|
||||||
return types.Float32(result), nil
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
package accounts
|
|
||||||
|
|
||||||
import (
|
|
||||||
_ "github.com/go-sql-driver/mysql"
|
|
||||||
_ "github.com/iwind/TeaGo/bootstrap"
|
|
||||||
)
|
|
||||||
@@ -976,7 +976,7 @@ func (this *NodeDAO) ComposeNodeConfig(tx *dbs.Tx, nodeId int64, cacheMap *utils
|
|||||||
}
|
}
|
||||||
|
|
||||||
for _, server := range servers {
|
for _, server := range servers {
|
||||||
serverConfig, err := SharedServerDAO.ComposeServerConfig(tx, server, cacheMap, true)
|
serverConfig, err := SharedServerDAO.ComposeServerConfig(tx, server, cacheMap, true, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1009,12 +1009,12 @@ func (this *ServerDAO) ComposeServerConfigWithServerId(tx *dbs.Tx, serverId int6
|
|||||||
if server == nil {
|
if server == nil {
|
||||||
return nil, ErrNotFound
|
return nil, ErrNotFound
|
||||||
}
|
}
|
||||||
return this.ComposeServerConfig(tx, server, nil, forNode)
|
return this.ComposeServerConfig(tx, server, nil, forNode, false)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ComposeServerConfig 构造服务的Config
|
// ComposeServerConfig 构造服务的Config
|
||||||
// forNode 是否是节点请求
|
// forNode 是否是节点请求
|
||||||
func (this *ServerDAO) ComposeServerConfig(tx *dbs.Tx, server *Server, cacheMap *utils.CacheMap, forNode bool) (*serverconfigs.ServerConfig, error) {
|
func (this *ServerDAO) ComposeServerConfig(tx *dbs.Tx, server *Server, cacheMap *utils.CacheMap, forNode bool, forList bool) (*serverconfigs.ServerConfig, error) {
|
||||||
if server == nil {
|
if server == nil {
|
||||||
return nil, ErrNotFound
|
return nil, ErrNotFound
|
||||||
}
|
}
|
||||||
@@ -1039,7 +1039,7 @@ func (this *ServerDAO) ComposeServerConfig(tx *dbs.Tx, server *Server, cacheMap
|
|||||||
|
|
||||||
var groupConfig *serverconfigs.ServerGroupConfig
|
var groupConfig *serverconfigs.ServerGroupConfig
|
||||||
for _, groupId := range server.DecodeGroupIds() {
|
for _, groupId := range server.DecodeGroupIds() {
|
||||||
groupConfig1, err := SharedServerGroupDAO.ComposeGroupConfig(tx, groupId, cacheMap)
|
groupConfig1, err := SharedServerGroupDAO.ComposeGroupConfig(tx, groupId, forList, cacheMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -1062,28 +1062,30 @@ func (this *ServerDAO) ComposeServerConfig(tx *dbs.Tx, server *Server, cacheMap
|
|||||||
}
|
}
|
||||||
|
|
||||||
// CNAME
|
// CNAME
|
||||||
config.SupportCNAME = server.SupportCNAME == 1
|
if !forList {
|
||||||
if server.ClusterId > 0 && len(server.DnsName) > 0 {
|
config.SupportCNAME = server.SupportCNAME == 1
|
||||||
clusterDNS, err := SharedNodeClusterDAO.FindClusterDNSInfo(tx, int64(server.ClusterId), cacheMap)
|
if server.ClusterId > 0 && len(server.DnsName) > 0 {
|
||||||
if err != nil {
|
clusterDNS, err := SharedNodeClusterDAO.FindClusterDNSInfo(tx, int64(server.ClusterId), cacheMap)
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if clusterDNS != nil && clusterDNS.DnsDomainId > 0 {
|
|
||||||
clusterDNSConfig, err := clusterDNS.DecodeDNSConfig()
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
if clusterDNS != nil && clusterDNS.DnsDomainId > 0 {
|
||||||
|
clusterDNSConfig, err := clusterDNS.DecodeDNSConfig()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
domain, err := dns.SharedDNSDomainDAO.FindEnabledDNSDomain(tx, int64(clusterDNS.DnsDomainId), cacheMap)
|
domain, err := dns.SharedDNSDomainDAO.FindEnabledDNSDomain(tx, int64(clusterDNS.DnsDomainId), cacheMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if domain != nil {
|
if domain != nil {
|
||||||
var cname = server.DnsName + "." + domain.Name
|
var cname = server.DnsName + "." + domain.Name
|
||||||
config.CNameDomain = cname
|
config.CNameDomain = cname
|
||||||
if clusterDNSConfig.CNAMEAsDomain {
|
if clusterDNSConfig.CNAMEAsDomain {
|
||||||
config.CNameAsDomain = true
|
config.CNameAsDomain = true
|
||||||
config.AliasServerNames = append(config.AliasServerNames, cname)
|
config.AliasServerNames = append(config.AliasServerNames, cname)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1174,61 +1176,71 @@ func (this *ServerDAO) ComposeServerConfig(tx *dbs.Tx, server *Server, cacheMap
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Web
|
// Web
|
||||||
if server.WebId > 0 {
|
if !forList {
|
||||||
webConfig, err := SharedHTTPWebDAO.ComposeWebConfig(tx, int64(server.WebId), cacheMap)
|
if server.WebId > 0 {
|
||||||
if err != nil {
|
webConfig, err := SharedHTTPWebDAO.ComposeWebConfig(tx, int64(server.WebId), cacheMap)
|
||||||
return nil, err
|
if err != nil {
|
||||||
}
|
return nil, err
|
||||||
if webConfig != nil {
|
}
|
||||||
config.Web = webConfig
|
if webConfig != nil {
|
||||||
|
config.Web = webConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ReverseProxy
|
// ReverseProxy
|
||||||
if IsNotNull(server.ReverseProxy) {
|
if !forList {
|
||||||
var reverseProxyRef = &serverconfigs.ReverseProxyRef{}
|
if IsNotNull(server.ReverseProxy) {
|
||||||
err := json.Unmarshal(server.ReverseProxy, reverseProxyRef)
|
var reverseProxyRef = &serverconfigs.ReverseProxyRef{}
|
||||||
if err != nil {
|
err := json.Unmarshal(server.ReverseProxy, reverseProxyRef)
|
||||||
return nil, err
|
if err != nil {
|
||||||
}
|
return nil, err
|
||||||
config.ReverseProxyRef = reverseProxyRef
|
}
|
||||||
|
config.ReverseProxyRef = reverseProxyRef
|
||||||
|
|
||||||
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if reverseProxyConfig != nil {
|
if reverseProxyConfig != nil {
|
||||||
config.ReverseProxy = reverseProxyConfig
|
config.ReverseProxy = reverseProxyConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// WAF策略
|
// WAF策略
|
||||||
var clusterId = int64(server.ClusterId)
|
var clusterId = int64(server.ClusterId)
|
||||||
httpFirewallPolicyId, err := SharedNodeClusterDAO.FindClusterHTTPFirewallPolicyId(tx, clusterId, cacheMap)
|
if !forList {
|
||||||
if err != nil {
|
httpFirewallPolicyId, err := SharedNodeClusterDAO.FindClusterHTTPFirewallPolicyId(tx, clusterId, cacheMap)
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if httpFirewallPolicyId > 0 {
|
|
||||||
config.HTTPFirewallPolicyId = httpFirewallPolicyId
|
|
||||||
}
|
|
||||||
|
|
||||||
// 缓存策略
|
|
||||||
httpCachePolicyId, err := SharedNodeClusterDAO.FindClusterHTTPCachePolicyId(tx, clusterId, cacheMap)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if httpCachePolicyId > 0 {
|
|
||||||
config.HTTPCachePolicyId = httpCachePolicyId
|
|
||||||
}
|
|
||||||
|
|
||||||
// traffic limit
|
|
||||||
if len(server.TrafficLimit) > 0 {
|
|
||||||
var trafficLimitConfig = &serverconfigs.TrafficLimitConfig{}
|
|
||||||
err = json.Unmarshal(server.TrafficLimit, trafficLimitConfig)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
config.TrafficLimit = trafficLimitConfig
|
if httpFirewallPolicyId > 0 {
|
||||||
|
config.HTTPFirewallPolicyId = httpFirewallPolicyId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 缓存策略
|
||||||
|
if !forList {
|
||||||
|
httpCachePolicyId, err := SharedNodeClusterDAO.FindClusterHTTPCachePolicyId(tx, clusterId, cacheMap)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if httpCachePolicyId > 0 {
|
||||||
|
config.HTTPCachePolicyId = httpCachePolicyId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// traffic limit
|
||||||
|
if !forList {
|
||||||
|
if len(server.TrafficLimit) > 0 {
|
||||||
|
var trafficLimitConfig = &serverconfigs.TrafficLimitConfig{}
|
||||||
|
err := json.Unmarshal(server.TrafficLimit, trafficLimitConfig)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
config.TrafficLimit = trafficLimitConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 用户套餐
|
// 用户套餐
|
||||||
@@ -1271,7 +1283,7 @@ func (this *ServerDAO) ComposeServerConfig(tx *dbs.Tx, server *Server, cacheMap
|
|||||||
if config.TrafficLimit != nil && config.TrafficLimit.IsOn && !config.TrafficLimit.IsEmpty() {
|
if config.TrafficLimit != nil && config.TrafficLimit.IsOn && !config.TrafficLimit.IsEmpty() {
|
||||||
if len(server.TrafficLimitStatus) > 0 {
|
if len(server.TrafficLimitStatus) > 0 {
|
||||||
var status = &serverconfigs.TrafficLimitStatus{}
|
var status = &serverconfigs.TrafficLimitStatus{}
|
||||||
err = json.Unmarshal(server.TrafficLimitStatus, status)
|
err := json.Unmarshal(server.TrafficLimitStatus, status)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -1282,14 +1294,16 @@ func (this *ServerDAO) ComposeServerConfig(tx *dbs.Tx, server *Server, cacheMap
|
|||||||
}
|
}
|
||||||
|
|
||||||
// UAM
|
// UAM
|
||||||
if teaconst.IsPlus && IsNotNull(server.Uam) {
|
if !forList {
|
||||||
var uamConfig = &serverconfigs.UAMConfig{}
|
if teaconst.IsPlus && IsNotNull(server.Uam) {
|
||||||
err = json.Unmarshal(server.Uam, uamConfig)
|
var uamConfig = &serverconfigs.UAMConfig{}
|
||||||
if err != nil {
|
err := json.Unmarshal(server.Uam, uamConfig)
|
||||||
return nil, err
|
if err != nil {
|
||||||
}
|
return nil, err
|
||||||
if uamConfig.IsOn {
|
}
|
||||||
config.UAM = uamConfig
|
if uamConfig.IsOn {
|
||||||
|
config.UAM = uamConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -279,7 +279,7 @@ func (this *ServerGroupDAO) InitGroupWeb(tx *dbs.Tx, groupId int64) (int64, erro
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ComposeGroupConfig 组合配置
|
// ComposeGroupConfig 组合配置
|
||||||
func (this *ServerGroupDAO) ComposeGroupConfig(tx *dbs.Tx, groupId int64, cacheMap *utils.CacheMap) (*serverconfigs.ServerGroupConfig, error) {
|
func (this *ServerGroupDAO) ComposeGroupConfig(tx *dbs.Tx, groupId int64, forList bool, cacheMap *utils.CacheMap) (*serverconfigs.ServerGroupConfig, error) {
|
||||||
if cacheMap == nil {
|
if cacheMap == nil {
|
||||||
cacheMap = utils.NewCacheMap()
|
cacheMap = utils.NewCacheMap()
|
||||||
}
|
}
|
||||||
@@ -315,65 +315,67 @@ func (this *ServerGroupDAO) ComposeGroupConfig(tx *dbs.Tx, groupId int64, cacheM
|
|||||||
IsOn: group.IsOn,
|
IsOn: group.IsOn,
|
||||||
}
|
}
|
||||||
|
|
||||||
if IsNotNull(group.HttpReverseProxy) {
|
if !forList {
|
||||||
reverseProxyRef := &serverconfigs.ReverseProxyRef{}
|
if IsNotNull(group.HttpReverseProxy) {
|
||||||
err := json.Unmarshal(group.HttpReverseProxy, reverseProxyRef)
|
reverseProxyRef := &serverconfigs.ReverseProxyRef{}
|
||||||
if err != nil {
|
err := json.Unmarshal(group.HttpReverseProxy, reverseProxyRef)
|
||||||
return nil, err
|
if err != nil {
|
||||||
}
|
return nil, err
|
||||||
config.HTTPReverseProxyRef = reverseProxyRef
|
}
|
||||||
|
config.HTTPReverseProxyRef = reverseProxyRef
|
||||||
|
|
||||||
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
}
|
||||||
|
if reverseProxyConfig != nil {
|
||||||
|
config.HTTPReverseProxy = reverseProxyConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if reverseProxyConfig != nil {
|
|
||||||
config.HTTPReverseProxy = reverseProxyConfig
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if IsNotNull(group.TcpReverseProxy) {
|
if IsNotNull(group.TcpReverseProxy) {
|
||||||
reverseProxyRef := &serverconfigs.ReverseProxyRef{}
|
reverseProxyRef := &serverconfigs.ReverseProxyRef{}
|
||||||
err := json.Unmarshal(group.TcpReverseProxy, reverseProxyRef)
|
err := json.Unmarshal(group.TcpReverseProxy, reverseProxyRef)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
config.TCPReverseProxyRef = reverseProxyRef
|
config.TCPReverseProxyRef = reverseProxyRef
|
||||||
|
|
||||||
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
}
|
||||||
|
if reverseProxyConfig != nil {
|
||||||
|
config.TCPReverseProxy = reverseProxyConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if reverseProxyConfig != nil {
|
|
||||||
config.TCPReverseProxy = reverseProxyConfig
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if IsNotNull(group.UdpReverseProxy) {
|
if IsNotNull(group.UdpReverseProxy) {
|
||||||
reverseProxyRef := &serverconfigs.ReverseProxyRef{}
|
reverseProxyRef := &serverconfigs.ReverseProxyRef{}
|
||||||
err := json.Unmarshal(group.UdpReverseProxy, reverseProxyRef)
|
err := json.Unmarshal(group.UdpReverseProxy, reverseProxyRef)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
config.UDPReverseProxyRef = reverseProxyRef
|
config.UDPReverseProxyRef = reverseProxyRef
|
||||||
|
|
||||||
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
reverseProxyConfig, err := SharedReverseProxyDAO.ComposeReverseProxyConfig(tx, reverseProxyRef.ReverseProxyId, cacheMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
}
|
||||||
|
if reverseProxyConfig != nil {
|
||||||
|
config.UDPReverseProxy = reverseProxyConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if reverseProxyConfig != nil {
|
|
||||||
config.UDPReverseProxy = reverseProxyConfig
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// web
|
// web
|
||||||
if group.WebId > 0 {
|
if group.WebId > 0 {
|
||||||
webConfig, err := SharedHTTPWebDAO.ComposeWebConfig(tx, int64(group.WebId), cacheMap)
|
webConfig, err := SharedHTTPWebDAO.ComposeWebConfig(tx, int64(group.WebId), cacheMap)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if webConfig != nil {
|
if webConfig != nil {
|
||||||
config.Web = webConfig
|
config.Web = webConfig
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
1
internal/installers/helpers/README.md
Normal file
1
internal/installers/helpers/README.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
远程安装依赖文件,单独放在一个目录防止安装包过大
|
||||||
91
internal/installers/helpers/unzip.go
Normal file
91
internal/installers/helpers/unzip.go
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
package helpers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"archive/zip"
|
||||||
|
"errors"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Unzip struct {
|
||||||
|
zipFile string
|
||||||
|
targetDir string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUnzip(zipFile string, targetDir string) *Unzip {
|
||||||
|
return &Unzip{
|
||||||
|
zipFile: zipFile,
|
||||||
|
targetDir: targetDir,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *Unzip) Run() error {
|
||||||
|
if len(this.zipFile) == 0 {
|
||||||
|
return errors.New("zip file should not be empty")
|
||||||
|
}
|
||||||
|
if len(this.targetDir) == 0 {
|
||||||
|
return errors.New("target dir should not be empty")
|
||||||
|
}
|
||||||
|
|
||||||
|
reader, err := zip.OpenReader(this.zipFile)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
_ = reader.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
for _, file := range reader.File {
|
||||||
|
info := file.FileInfo()
|
||||||
|
target := this.targetDir + "/" + file.Name
|
||||||
|
|
||||||
|
// 目录
|
||||||
|
if info.IsDir() {
|
||||||
|
stat, err := os.Stat(target)
|
||||||
|
if err != nil {
|
||||||
|
if !os.IsNotExist(err) {
|
||||||
|
return err
|
||||||
|
} else {
|
||||||
|
err = os.MkdirAll(target, info.Mode())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if !stat.IsDir() {
|
||||||
|
err = os.MkdirAll(target, info.Mode())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// 文件
|
||||||
|
err := func(file *zip.File, target string) error {
|
||||||
|
fileReader, err := file.Open()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
_ = fileReader.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
fileWriter, err := os.OpenFile(target, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, file.FileInfo().Mode())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
_ = fileWriter.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
_, err = io.Copy(fileWriter, fileReader)
|
||||||
|
return err
|
||||||
|
}(file, target)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
17
internal/installers/helpers/unzip_test.go
Normal file
17
internal/installers/helpers/unzip_test.go
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
package helpers_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/TeaOSLab/EdgeAPI/internal/installers/helpers"
|
||||||
|
"github.com/iwind/TeaGo/Tea"
|
||||||
|
_ "github.com/iwind/TeaGo/bootstrap"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestUnzip_Run(t *testing.T) {
|
||||||
|
var unzip = helpers.NewUnzip(Tea.Root+"/deploy/edge-node-v0.0.1.zip", Tea.Root+"/deploy/")
|
||||||
|
err := unzip.Run()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
t.Log("OK")
|
||||||
|
}
|
||||||
@@ -147,7 +147,7 @@ func (this *BaseInstaller) LookupLatestInstaller(filePrefix string) (string, err
|
|||||||
|
|
||||||
// InstallHelper 上传安装助手
|
// InstallHelper 上传安装助手
|
||||||
func (this *BaseInstaller) InstallHelper(targetDir string, role nodeconfigs.NodeRole) (env *Env, err error) {
|
func (this *BaseInstaller) InstallHelper(targetDir string, role nodeconfigs.NodeRole) (env *Env, err error) {
|
||||||
uname, _, err := this.client.Exec("uname -a")
|
uname, _, err := this.client.Exec("/usr/bin/uname -a")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return env, err
|
return env, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -366,7 +366,7 @@ func (this *NodeQueue) StopNode(nodeId int64) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 我们先尝试Systemd停止
|
// 我们先尝试Systemd停止
|
||||||
_, _, _ = installer.client.Exec("systemctl stop edge-node")
|
_, _, _ = installer.client.Exec("/usr/bin/systemctl stop edge-node")
|
||||||
|
|
||||||
// 执行stop
|
// 执行stop
|
||||||
_, stderr, err := installer.client.Exec(exe + " stop")
|
_, stderr, err := installer.client.Exec(exe + " stop")
|
||||||
|
|||||||
@@ -357,16 +357,6 @@ func (this *APINode) registerServices(server *grpc.Server) {
|
|||||||
pb.RegisterServerDailyStatServiceServer(server, instance)
|
pb.RegisterServerDailyStatServiceServer(server, instance)
|
||||||
this.rest(instance)
|
this.rest(instance)
|
||||||
}
|
}
|
||||||
{
|
|
||||||
var instance = this.serviceInstance(&services.UserBillService{}).(*services.UserBillService)
|
|
||||||
pb.RegisterUserBillServiceServer(server, instance)
|
|
||||||
this.rest(instance)
|
|
||||||
}
|
|
||||||
{
|
|
||||||
var instance = this.serviceInstance(&services.ServerBillService{}).(*services.ServerBillService)
|
|
||||||
pb.RegisterServerBillServiceServer(server, instance)
|
|
||||||
this.rest(instance)
|
|
||||||
}
|
|
||||||
{
|
{
|
||||||
var instance = this.serviceInstance(&services.LoginService{}).(*services.LoginService)
|
var instance = this.serviceInstance(&services.LoginService{}).(*services.LoginService)
|
||||||
pb.RegisterLoginServiceServer(server, instance)
|
pb.RegisterLoginServiceServer(server, instance)
|
||||||
|
|||||||
@@ -840,21 +840,38 @@ func (this *ServerService) ListEnabledServersMatch(ctx context.Context, req *pb.
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 配置
|
// 配置
|
||||||
config, err := models.SharedServerDAO.ComposeServerConfig(tx, server, nil, false)
|
config, err := models.SharedServerDAO.ComposeServerConfig(tx, server, nil, false, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
var countServerNames int32 = 0
|
||||||
|
for _, serverName := range config.ServerNames {
|
||||||
|
if len(serverName.SubNames) > 0 {
|
||||||
|
countServerNames += int32(len(serverName.SubNames))
|
||||||
|
} else {
|
||||||
|
countServerNames++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if req.IgnoreServerNames && len(config.ServerNames) > 0 {
|
||||||
|
config.ServerNames = config.ServerNames[:1]
|
||||||
|
}
|
||||||
configJSON, err := json.Marshal(config)
|
configJSON, err := json.Marshal(config)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 忽略信息
|
||||||
|
if req.IgnoreServerNames {
|
||||||
|
server.ServerNames = nil
|
||||||
|
}
|
||||||
|
|
||||||
result = append(result, &pb.Server{
|
result = append(result, &pb.Server{
|
||||||
Id: int64(server.Id),
|
Id: int64(server.Id),
|
||||||
IsOn: server.IsOn,
|
IsOn: server.IsOn,
|
||||||
Type: server.Type,
|
Type: server.Type,
|
||||||
Config: configJSON,
|
Config: configJSON,
|
||||||
Name: server.Name,
|
Name: server.Name,
|
||||||
|
CountServerNames: countServerNames,
|
||||||
Description: server.Description,
|
Description: server.Description,
|
||||||
HttpJSON: server.Http,
|
HttpJSON: server.Http,
|
||||||
HttpsJSON: server.Https,
|
HttpsJSON: server.Https,
|
||||||
@@ -986,7 +1003,7 @@ func (this *ServerService) FindEnabledServer(ctx context.Context, req *pb.FindEn
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 配置
|
// 配置
|
||||||
config, err := models.SharedServerDAO.ComposeServerConfig(tx, server, nil, userId > 0)
|
config, err := models.SharedServerDAO.ComposeServerConfig(tx, server, nil, userId > 0, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,132 +0,0 @@
|
|||||||
// Copyright 2022 Liuxiangchao iwind.liu@gmail.com. All rights reserved.
|
|
||||||
|
|
||||||
package services
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/db/models"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/utils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
|
||||||
)
|
|
||||||
|
|
||||||
// ServerBillService 服务账单相关服务
|
|
||||||
type ServerBillService struct {
|
|
||||||
BaseService
|
|
||||||
}
|
|
||||||
|
|
||||||
// CountAllServerBills 查询服务账单数量
|
|
||||||
func (this *ServerBillService) CountAllServerBills(ctx context.Context, req *pb.CountAllServerBillsRequest) (*pb.RPCCountResponse, error) {
|
|
||||||
_, userId, err := this.ValidateAdminAndUser(ctx, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if userId > 0 {
|
|
||||||
req.UserId = userId
|
|
||||||
}
|
|
||||||
|
|
||||||
var tx = this.NullTx()
|
|
||||||
count, err := models.SharedServerBillDAO.CountServerBills(tx, req.UserId, req.Month)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return this.SuccessCount(count)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListServerBills 查询服务账单列表
|
|
||||||
func (this *ServerBillService) ListServerBills(ctx context.Context, req *pb.ListServerBillsRequest) (*pb.ListServerBillsResponse, error) {
|
|
||||||
_, userId, err := this.ValidateAdminAndUser(ctx, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if userId > 0 {
|
|
||||||
req.UserId = userId
|
|
||||||
}
|
|
||||||
|
|
||||||
var tx = this.NullTx()
|
|
||||||
serverBills, err := models.SharedServerBillDAO.ListServerBills(tx, req.UserId, req.Month, req.Offset, req.Size)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var pbServerBills = []*pb.ServerBill{}
|
|
||||||
var cacheMap = utils.NewCacheMap()
|
|
||||||
for _, bill := range serverBills {
|
|
||||||
// user
|
|
||||||
user, err := models.SharedUserDAO.FindBasicUserWithoutState(tx, int64(bill.UserId))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
var pbUser = &pb.User{Id: int64(bill.UserId)}
|
|
||||||
if user != nil {
|
|
||||||
pbUser = &pb.User{
|
|
||||||
Id: int64(bill.UserId),
|
|
||||||
Username: user.Username,
|
|
||||||
Fullname: user.Fullname,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// plan
|
|
||||||
var pbPlan *pb.Plan
|
|
||||||
if bill.PlanId > 0 {
|
|
||||||
plan, err := models.SharedPlanDAO.FindEnabledPlan(tx, int64(bill.PlanId))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if plan != nil {
|
|
||||||
pbPlan = &pb.Plan{
|
|
||||||
Id: int64(plan.Id),
|
|
||||||
Name: plan.Name,
|
|
||||||
PriceType: plan.PriceType,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// user plan
|
|
||||||
var pbUserPlan *pb.UserPlan
|
|
||||||
if bill.UserPlanId > 0 {
|
|
||||||
userPlan, err := models.SharedUserPlanDAO.FindEnabledUserPlan(tx, int64(bill.UserPlanId), cacheMap)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if userPlan != nil {
|
|
||||||
pbUserPlan = &pb.UserPlan{
|
|
||||||
Id: int64(userPlan.Id),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// server
|
|
||||||
var pbServer *pb.Server
|
|
||||||
if bill.ServerId > 0 {
|
|
||||||
server, err := models.SharedServerDAO.FindEnabledServerBasic(tx, int64(bill.ServerId))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if server != nil {
|
|
||||||
pbServer = &pb.Server{Id: int64(bill.ServerId), Name: server.Name}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pbServerBills = append(pbServerBills, &pb.ServerBill{
|
|
||||||
Id: int64(bill.Id),
|
|
||||||
UserId: int64(bill.UserId),
|
|
||||||
ServerId: int64(bill.ServerId),
|
|
||||||
Amount: float32(bill.Amount),
|
|
||||||
PriceType: bill.PriceType,
|
|
||||||
CreatedAt: int64(bill.CreatedAt),
|
|
||||||
UserPlanId: int64(bill.UserPlanId),
|
|
||||||
PlanId: int64(bill.PlanId),
|
|
||||||
TotalTrafficBytes: int64(bill.TotalTrafficBytes),
|
|
||||||
BandwidthPercentileBytes: int64(bill.BandwidthPercentileBytes),
|
|
||||||
BandwidthPercentile: int32(bill.BandwidthPercentile),
|
|
||||||
User: pbUser,
|
|
||||||
Plan: pbPlan,
|
|
||||||
UserPlan: pbUserPlan,
|
|
||||||
Server: pbServer,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return &pb.ListServerBillsResponse{ServerBills: pbServerBills}, nil
|
|
||||||
}
|
|
||||||
@@ -463,7 +463,7 @@ func (this *ServerGroupService) FindEnabledServerGroupConfigInfo(ctx context.Con
|
|||||||
result.HasUDPReverseProxy = ref.IsPrior
|
result.HasUDPReverseProxy = ref.IsPrior
|
||||||
}
|
}
|
||||||
|
|
||||||
config, err := models.SharedServerGroupDAO.ComposeGroupConfig(tx, int64(group.Id), nil)
|
config, err := models.SharedServerGroupDAO.ComposeGroupConfig(tx, int64(group.Id), false, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,255 +0,0 @@
|
|||||||
package services
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/db/models"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/db/models/accounts"
|
|
||||||
"github.com/TeaOSLab/EdgeAPI/internal/errors"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/userconfigs"
|
|
||||||
"github.com/iwind/TeaGo/dbs"
|
|
||||||
"github.com/iwind/TeaGo/maps"
|
|
||||||
timeutil "github.com/iwind/TeaGo/utils/time"
|
|
||||||
"regexp"
|
|
||||||
)
|
|
||||||
|
|
||||||
// UserBillService 账单相关服务
|
|
||||||
type UserBillService struct {
|
|
||||||
BaseService
|
|
||||||
}
|
|
||||||
|
|
||||||
// GenerateAllUserBills 手工生成订单
|
|
||||||
func (this *UserBillService) GenerateAllUserBills(ctx context.Context, req *pb.GenerateAllUserBillsRequest) (*pb.RPCSuccess, error) {
|
|
||||||
_, err := this.ValidateAdmin(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// 校验Month
|
|
||||||
if !regexp.MustCompile(`^\d{6}$`).MatchString(req.Month) {
|
|
||||||
return nil, errors.New("invalid month '" + req.Month + "'")
|
|
||||||
}
|
|
||||||
if req.Month >= timeutil.Format("Ym") {
|
|
||||||
return nil, errors.New("invalid month '" + req.Month + "'")
|
|
||||||
}
|
|
||||||
|
|
||||||
var tx = this.NullTx()
|
|
||||||
|
|
||||||
err = models.SharedUserBillDAO.GenerateBills(tx, req.Month)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.Success()
|
|
||||||
}
|
|
||||||
|
|
||||||
// CountAllUserBills 计算所有账单数量
|
|
||||||
func (this *UserBillService) CountAllUserBills(ctx context.Context, req *pb.CountAllUserBillsRequest) (*pb.RPCCountResponse, error) {
|
|
||||||
_, userId, err := this.ValidateAdminAndUser(ctx, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var tx = this.NullTx()
|
|
||||||
if userId > 0 {
|
|
||||||
req.UserId = userId
|
|
||||||
}
|
|
||||||
|
|
||||||
count, err := models.SharedUserBillDAO.CountAllUserBills(tx, req.PaidFlag, req.UserId, req.Month)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return this.SuccessCount(count)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListUserBills 列出单页账单
|
|
||||||
func (this *UserBillService) ListUserBills(ctx context.Context, req *pb.ListUserBillsRequest) (*pb.ListUserBillsResponse, error) {
|
|
||||||
_, userId, err := this.ValidateAdminAndUser(ctx, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var tx = this.NullTx()
|
|
||||||
if userId > 0 {
|
|
||||||
req.UserId = userId
|
|
||||||
}
|
|
||||||
|
|
||||||
bills, err := models.SharedUserBillDAO.ListUserBills(tx, req.PaidFlag, req.UserId, req.Month, req.Offset, req.Size)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
result := []*pb.UserBill{}
|
|
||||||
for _, bill := range bills {
|
|
||||||
user, err := models.SharedUserDAO.FindBasicUserWithoutState(tx, int64(bill.UserId))
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if user == nil {
|
|
||||||
user = &models.User{Id: bill.UserId}
|
|
||||||
}
|
|
||||||
|
|
||||||
result = append(result, &pb.UserBill{
|
|
||||||
Id: int64(bill.Id),
|
|
||||||
User: &pb.User{
|
|
||||||
Id: int64(bill.UserId),
|
|
||||||
Fullname: user.Fullname,
|
|
||||||
Username: user.Username,
|
|
||||||
IsDeleted: user.State == models.UserStateDisabled,
|
|
||||||
},
|
|
||||||
Type: bill.Type,
|
|
||||||
TypeName: models.SharedUserBillDAO.BillTypeName(bill.Type),
|
|
||||||
Description: bill.Description,
|
|
||||||
Amount: float32(bill.Amount),
|
|
||||||
Month: bill.Month,
|
|
||||||
CanPay: bill.CanPay,
|
|
||||||
IsPaid: bill.IsPaid,
|
|
||||||
PaidAt: int64(bill.PaidAt),
|
|
||||||
Code: bill.Code,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return &pb.ListUserBillsResponse{UserBills: result}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// FindUserBill 查找账单信息
|
|
||||||
func (this *UserBillService) FindUserBill(ctx context.Context, req *pb.FindUserBillRequest) (*pb.FindUserBillResponse, error) {
|
|
||||||
_, userId, err := this.ValidateAdminAndUser(ctx, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var tx = this.NullTx()
|
|
||||||
|
|
||||||
// 检查用户
|
|
||||||
if userId > 0 {
|
|
||||||
err = models.SharedUserBillDAO.CheckUserBill(tx, userId, req.UserBillId)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bill, err := models.SharedUserBillDAO.FindUserBill(tx, req.UserBillId)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
if bill == nil {
|
|
||||||
return &pb.FindUserBillResponse{UserBill: nil}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// 用户
|
|
||||||
var pbUser = &pb.User{Id: int64(bill.UserId)}
|
|
||||||
user, err := models.SharedUserDAO.FindEnabledUser(tx, int64(bill.UserId), nil)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if user != nil {
|
|
||||||
pbUser = &pb.User{
|
|
||||||
Id: int64(user.Id),
|
|
||||||
Username: user.Username,
|
|
||||||
Fullname: user.Fullname,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return &pb.FindUserBillResponse{
|
|
||||||
UserBill: &pb.UserBill{
|
|
||||||
Id: int64(bill.Id),
|
|
||||||
User: pbUser,
|
|
||||||
Type: bill.Type,
|
|
||||||
TypeName: models.SharedUserBillDAO.BillTypeName(bill.Type),
|
|
||||||
Description: bill.Description,
|
|
||||||
Amount: float32(bill.Amount),
|
|
||||||
Month: bill.Month,
|
|
||||||
CanPay: bill.CanPay,
|
|
||||||
IsPaid: bill.IsPaid,
|
|
||||||
PaidAt: int64(bill.PaidAt),
|
|
||||||
Code: bill.Code,
|
|
||||||
},
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// PayUserBill 支付账单
|
|
||||||
func (this *UserBillService) PayUserBill(ctx context.Context, req *pb.PayUserBillRequest) (*pb.RPCSuccess, error) {
|
|
||||||
_, userId, err := this.ValidateAdminAndUser(ctx, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
err = this.RunTx(func(tx *dbs.Tx) error {
|
|
||||||
// 检查用户
|
|
||||||
if userId > 0 {
|
|
||||||
err = models.SharedUserBillDAO.CheckUserBill(tx, userId, req.UserBillId)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 是否存在
|
|
||||||
bill, err := models.SharedUserBillDAO.FindUserBill(tx, req.UserBillId)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if bill == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
userId = int64(bill.UserId)
|
|
||||||
|
|
||||||
// 是否已支付
|
|
||||||
if bill.IsPaid {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if bill.Amount <= 0 {
|
|
||||||
// 直接修改为已支付
|
|
||||||
return models.SharedUserBillDAO.UpdateUserBillIsPaid(tx, req.UserBillId, true)
|
|
||||||
}
|
|
||||||
|
|
||||||
if !bill.CanPay {
|
|
||||||
return errors.New("can not pay now")
|
|
||||||
}
|
|
||||||
|
|
||||||
// 余额是否足够
|
|
||||||
account, err := accounts.SharedUserAccountDAO.FindUserAccountWithUserId(tx, userId)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if account == nil {
|
|
||||||
return errors.New("can not find user account")
|
|
||||||
}
|
|
||||||
|
|
||||||
if account.Total < bill.Amount {
|
|
||||||
return errors.New("not enough balance to pay")
|
|
||||||
}
|
|
||||||
|
|
||||||
err = accounts.SharedUserAccountDAO.UpdateUserAccount(tx, int64(account.Id), -float32(bill.Amount), userconfigs.AccountEventTypePayBill, "支付账单"+bill.Code, maps.Map{"billId": bill.Id})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// 修改为已支付
|
|
||||||
return models.SharedUserBillDAO.UpdateUserBillIsPaid(tx, req.UserBillId, true)
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.Success()
|
|
||||||
}
|
|
||||||
|
|
||||||
// SumUserUnpaidBills 计算用户所有未支付账单总额
|
|
||||||
func (this *UserBillService) SumUserUnpaidBills(ctx context.Context, req *pb.SumUserUnpaidBillsRequest) (*pb.SumUserUnpaidBillsResponse, error) {
|
|
||||||
_, userId, err := this.ValidateAdminAndUser(ctx, true)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var tx = this.NullTx()
|
|
||||||
if userId > 0 {
|
|
||||||
req.UserId = userId
|
|
||||||
}
|
|
||||||
|
|
||||||
sum, err := models.SharedUserBillDAO.SumUnpaidUserBill(tx, userId)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return &pb.SumUserUnpaidBillsResponse{Amount: sum}, nil
|
|
||||||
}
|
|
||||||
File diff suppressed because one or more lines are too long
Reference in New Issue
Block a user