server/pkg/deduction/deduction.go

375 lines
11 KiB
Go

// Package deduction provides functionality for calculating remaining amounts
// in subscription billing systems, supporting various time units and traffic-based calculations.
package deduction
import (
"errors"
"fmt"
"math"
"time"
"github.com/perfect-panel/server/pkg/tool"
)
const (
// Time unit constants for subscription billing
UnitTimeNoLimit = "NoLimit" // Unlimited time subscription
UnitTimeYear = "Year" // Annual subscription
UnitTimeMonth = "Month" // Monthly subscription
UnitTimeDay = "Day" // Daily subscription
UnitTimeHour = "Hour" // Hourly subscription
UnitTimeMinute = "Minute" // Per-minute subscription
// Reset cycle constants for traffic resets
ResetCycleNone = 0 // No reset cycle
ResetCycle1st = 1 // Reset on 1st of each month
ResetCycleMonthly = 2 // Reset monthly based on start date
ResetCycleYear = 3 // Reset yearly based on start date
// Safety limits for overflow protection
maxInt64 = math.MaxInt64
minInt64 = math.MinInt64
)
// Error definitions for validation and calculation failures
var (
ErrInvalidQuantity = errors.New("order quantity cannot be zero or negative")
ErrInvalidAmount = errors.New("order amount cannot be negative")
ErrInvalidTraffic = errors.New("traffic values cannot be negative")
ErrInvalidTimeRange = errors.New("expire time must be after start time")
ErrInvalidUnitTime = errors.New("invalid unit time")
ErrInvalidDeductionRatio = errors.New("deduction ratio must be between 0 and 100")
ErrOverflow = errors.New("calculation overflow")
)
// Subscribe represents a subscription with time and traffic limits
type Subscribe struct {
StartTime time.Time // Subscription start time
ExpireTime time.Time // Subscription expiration time
Traffic int64 // Total traffic allowance in bytes
Download int64 // Downloaded traffic in bytes
Upload int64 // Uploaded traffic in bytes
UnitTime string // Time unit for billing (Year, Month, Day, etc.)
UnitPrice int64 // Price per unit time
ResetCycle int64 // Traffic reset cycle
DeductionRatio int64 // Deduction ratio for weighted calculations (0-100)
}
// Order represents a purchase order for subscription calculation
type Order struct {
Amount int64 // Total order amount
Quantity int64 // Order quantity
}
// Validate checks if the Subscribe struct contains valid data
func (s *Subscribe) Validate() error {
if s.Traffic < 0 || s.Download < 0 || s.Upload < 0 {
return ErrInvalidTraffic
}
if s.Download+s.Upload > s.Traffic {
return fmt.Errorf("download + upload (%d) cannot exceed total traffic (%d)", s.Download+s.Upload, s.Traffic)
}
if !s.ExpireTime.After(s.StartTime) {
return ErrInvalidTimeRange
}
if s.DeductionRatio < 0 || s.DeductionRatio > 100 {
return ErrInvalidDeductionRatio
}
validUnitTimes := []string{UnitTimeNoLimit, UnitTimeYear, UnitTimeMonth, UnitTimeDay, UnitTimeHour, UnitTimeMinute}
valid := false
for _, ut := range validUnitTimes {
if s.UnitTime == ut {
valid = true
break
}
}
if !valid {
return ErrInvalidUnitTime
}
return nil
}
// Validate checks if the Order struct contains valid data
func (o *Order) Validate() error {
if o.Quantity <= 0 {
return ErrInvalidQuantity
}
if o.Amount < 0 {
return ErrInvalidAmount
}
return nil
}
// safeMultiply performs multiplication with overflow protection
func safeMultiply(a, b int64) (int64, error) {
if a == 0 || b == 0 {
return 0, nil
}
if a > 0 && b > 0 {
if a > maxInt64/b {
return 0, ErrOverflow
}
} else if a < 0 && b < 0 {
if a < maxInt64/b {
return 0, ErrOverflow
}
} else {
if (a > 0 && b < minInt64/a) || (a < 0 && b > minInt64/a) {
return 0, ErrOverflow
}
}
return a * b, nil
}
// safeAdd performs addition with overflow protection
func safeAdd(a, b int64) (int64, error) {
if (b > 0 && a > maxInt64-b) || (b < 0 && a < minInt64-b) {
return 0, ErrOverflow
}
return a + b, nil
}
// safeDivide performs division with zero-division protection
func safeDivide(a, b int64) (int64, error) {
if b == 0 {
return 0, errors.New("division by zero")
}
return a / b, nil
}
// CalculateRemainingAmount calculates the remaining refund amount for a subscription
// based on unused time and traffic. Returns the amount and any calculation errors.
func CalculateRemainingAmount(sub Subscribe, order Order) (int64, error) {
if err := sub.Validate(); err != nil {
return 0, fmt.Errorf("invalid subscription: %w", err)
}
if err := order.Validate(); err != nil {
return 0, fmt.Errorf("invalid order: %w", err)
}
if sub.UnitTime == UnitTimeNoLimit && sub.ResetCycle != 0 {
return 0, nil
}
unitPrice, err := safeDivide(order.Amount, order.Quantity)
if err != nil {
return 0, fmt.Errorf("failed to calculate unit price: %w", err)
}
sub.UnitPrice = unitPrice
loc, err := time.LoadLocation(sub.StartTime.Location().String())
if err != nil {
loc = time.UTC
}
now := time.Now().In(loc)
switch sub.UnitTime {
case UnitTimeNoLimit:
return calculateNoLimitAmount(sub, order)
case UnitTimeYear:
remainingYears := tool.YearDiff(now, sub.ExpireTime)
remainingUnitTimeAmount, err := calculateRemainingUnitTimeAmount(sub)
if err != nil {
return 0, err
}
yearAmount, err := safeMultiply(int64(remainingYears), sub.UnitPrice)
if err != nil {
return 0, fmt.Errorf("year calculation overflow: %w", err)
}
total, err := safeAdd(yearAmount, remainingUnitTimeAmount)
if err != nil {
return 0, fmt.Errorf("total calculation overflow: %w", err)
}
return total, nil
case UnitTimeMonth:
remainingMonths := tool.MonthDiff(now, sub.ExpireTime)
remainingUnitTimeAmount, err := calculateRemainingUnitTimeAmount(sub)
if err != nil {
return 0, err
}
monthAmount, err := safeMultiply(int64(remainingMonths), sub.UnitPrice)
if err != nil {
return 0, fmt.Errorf("month calculation overflow: %w", err)
}
total, err := safeAdd(monthAmount, remainingUnitTimeAmount)
if err != nil {
return 0, fmt.Errorf("total calculation overflow: %w", err)
}
return total, nil
case UnitTimeDay:
remainingDays := tool.DayDiff(now, sub.ExpireTime)
remainingUnitTimeAmount, err := calculateRemainingUnitTimeAmount(sub)
if err != nil {
return 0, err
}
dayAmount, err := safeMultiply(remainingDays, sub.UnitPrice)
if err != nil {
return 0, fmt.Errorf("day calculation overflow: %w", err)
}
total, err := safeAdd(dayAmount, remainingUnitTimeAmount)
if err != nil {
return 0, fmt.Errorf("total calculation overflow: %w", err)
}
return total, nil
}
return 0, nil
}
// calculateNoLimitAmount calculates refund amount for unlimited time subscriptions
// based on unused traffic only
func calculateNoLimitAmount(sub Subscribe, order Order) (int64, error) {
if sub.Traffic == 0 {
return 0, nil
}
usedTraffic := sub.Traffic - sub.Download - sub.Upload
if usedTraffic < 0 {
usedTraffic = 0
}
unitPrice := float64(order.Amount) / float64(sub.Traffic)
result := float64(usedTraffic) * unitPrice
if result > float64(maxInt64) || result < float64(minInt64) {
return 0, ErrOverflow
}
return int64(result), nil
}
// calculateRemainingUnitTimeAmount calculates the remaining amount based on
// both time and traffic usage, applying deduction ratios when specified
func calculateRemainingUnitTimeAmount(sub Subscribe) (int64, error) {
now := time.Now()
trafficWeight, timeWeight := calculateWeights(sub.DeductionRatio)
remainingDays, totalDays := getRemainingAndTotalDays(sub, now)
if totalDays == 0 {
return 0, nil
}
remainingTraffic := sub.Traffic - sub.Download - sub.Upload
if remainingTraffic < 0 {
remainingTraffic = 0
}
remainingTimeAmount, err := calculateProportionalAmount(sub.UnitPrice, remainingDays, totalDays)
if err != nil {
return 0, fmt.Errorf("time amount calculation failed: %w", err)
}
if sub.Traffic == 0 {
return remainingTimeAmount, nil
}
remainingTrafficAmount, err := calculateProportionalAmount(sub.UnitPrice, remainingTraffic, sub.Traffic)
if err != nil {
return 0, fmt.Errorf("traffic amount calculation failed: %w", err)
}
if sub.DeductionRatio != 0 {
return calculateWeightedAmount(sub.UnitPrice, remainingTraffic, sub.Traffic, remainingDays, totalDays, trafficWeight, timeWeight)
}
return min(remainingTimeAmount, remainingTrafficAmount), nil
}
// calculateWeights converts deduction ratio to traffic and time weights
// for weighted calculations
func calculateWeights(deductionRatio int64) (float64, float64) {
if deductionRatio == 0 {
return 0, 0
}
trafficWeight := float64(deductionRatio) / 100
timeWeight := 1 - trafficWeight
return trafficWeight, timeWeight
}
// getRemainingAndTotalDays calculates remaining and total days based on
// the subscription's reset cycle configuration
func getRemainingAndTotalDays(sub Subscribe, now time.Time) (int64, int64) {
switch sub.ResetCycle {
case ResetCycleNone:
remaining := sub.ExpireTime.Sub(now).Hours() / 24
total := sub.ExpireTime.Sub(sub.StartTime).Hours() / 24
if remaining < 0 {
remaining = 0
}
if total < 0 {
total = 0
}
return int64(remaining), int64(total)
case ResetCycle1st:
return tool.DaysToNextMonth(now), tool.GetLastDayOfMonth(now)
case ResetCycleMonthly:
remaining := tool.DaysToMonthDay(now, sub.StartTime.Day()) - 1
total := tool.DaysToMonthDay(now, sub.StartTime.Day())
if remaining < 0 {
remaining = 0
}
return remaining, total
case ResetCycleYear:
return tool.DaysToYearDay(now, int(sub.StartTime.Month()), sub.StartTime.Day()),
tool.GetYearDays(now, int(sub.StartTime.Month()), sub.StartTime.Day())
}
return 0, 0
}
// calculateWeightedAmount applies weighted calculation combining both time and traffic
// remaining ratios based on the specified weights
func calculateWeightedAmount(unitPrice, remainingTraffic, totalTraffic, remainingDays, totalDays int64, trafficWeight, timeWeight float64) (int64, error) {
if totalDays == 0 || totalTraffic == 0 {
return 0, nil
}
remainingTimeRatio := float64(remainingDays) / float64(totalDays)
remainingTrafficRatio := float64(remainingTraffic) / float64(totalTraffic)
weightedRemainingRatio := (timeWeight * remainingTimeRatio) + (trafficWeight * remainingTrafficRatio)
result := float64(unitPrice) * weightedRemainingRatio
if result > float64(maxInt64) || result < float64(minInt64) {
return 0, ErrOverflow
}
return int64(result), nil
}
// calculateProportionalAmount calculates proportional amount based on
// remaining vs total ratio with overflow protection
func calculateProportionalAmount(unitPrice, remaining, total int64) (int64, error) {
if total == 0 {
return 0, nil
}
result := float64(unitPrice) * (float64(remaining) / float64(total))
if result > float64(maxInt64) || result < float64(minInt64) {
return 0, ErrOverflow
}
return int64(result), nil
}