package repository import ( "context" "fmt" dbgen "github.com/SamuelTariku/FortuneBet-Backend/gen/db" "github.com/SamuelTariku/FortuneBet-Backend/internal/domain" "github.com/jackc/pgx/v5/pgtype" ) func convertDBTransaction(transaction dbgen.Transaction) domain.Transaction { return domain.Transaction{ ID: transaction.ID, Amount: domain.Currency(transaction.Amount), BranchID: transaction.BranchID, CashierID: transaction.CashierID, BetID: transaction.BetID, NumberOfOutcomes: transaction.NumberOfOutcomes, Type: domain.TransactionType(transaction.Type), PaymentOption: domain.PaymentOption(transaction.PaymentOption), FullName: transaction.FullName, PhoneNumber: transaction.PhoneNumber, BankCode: transaction.BankCode, BeneficiaryName: transaction.BeneficiaryName, AccountName: transaction.AccountName, AccountNumber: transaction.AccountNumber, ReferenceNumber: transaction.ReferenceNumber, ApprovedBy: domain.ValidInt64{ Value: transaction.ApprovedBy.Int64, Valid: transaction.ApprovedBy.Valid, }, CreatedAt: transaction.CreatedAt.Time, UpdatedAt: transaction.UpdatedAt.Time, Verified: transaction.Verified, BranchName: transaction.BranchName, BranchLocation: transaction.BranchLocation, CashierName: transaction.CashierName, CompanyID: transaction.CompanyID, ApproverName: domain.ValidString{ Value: transaction.ApproverName.String, Valid: transaction.ApprovedBy.Valid, }, } } func convertCreateTransaction(transaction domain.CreateTransaction) dbgen.CreateTransactionParams { return dbgen.CreateTransactionParams{ Amount: int64(transaction.Amount), BranchID: transaction.BranchID, CashierID: transaction.CashierID, BetID: transaction.BetID, Type: int64(transaction.Type), PaymentOption: int64(transaction.PaymentOption), FullName: transaction.FullName, PhoneNumber: transaction.PhoneNumber, BankCode: transaction.BankCode, BeneficiaryName: transaction.BeneficiaryName, AccountName: transaction.AccountName, AccountNumber: transaction.AccountNumber, ReferenceNumber: transaction.ReferenceNumber, NumberOfOutcomes: transaction.NumberOfOutcomes, BranchName: transaction.BranchName, BranchLocation: transaction.BranchLocation, CashierName: transaction.CashierName, CompanyID: transaction.CompanyID, } } func (s *Store) CreateTransaction(ctx context.Context, transaction domain.CreateTransaction) (domain.Transaction, error) { newTransaction, err := s.queries.CreateTransaction(ctx, convertCreateTransaction(transaction)) if err != nil { return domain.Transaction{}, err } return convertDBTransaction(newTransaction), err } func (s *Store) GetTransactionByID(ctx context.Context, id int64) (domain.Transaction, error) { transaction, err := s.queries.GetTransactionByID(ctx, id) if err != nil { return domain.Transaction{}, err } return convertDBTransaction(transaction), nil } func (s *Store) GetAllTransactions(ctx context.Context, filter domain.TransactionFilter) ([]domain.Transaction, error) { transaction, err := s.queries.GetAllTransactions(ctx, dbgen.GetAllTransactionsParams{ BranchID: pgtype.Int8{ Int64: filter.BranchID.Value, Valid: filter.BranchID.Valid, }, CompanyID: pgtype.Int8{ Int64: filter.CompanyID.Value, Valid: filter.CompanyID.Valid, }, CashierID: pgtype.Int8{ Int64: filter.CashierID.Value, Valid: filter.CashierID.Valid, }, }) if err != nil { return nil, err } var result []domain.Transaction = make([]domain.Transaction, 0, len(transaction)) for _, ticket := range transaction { result = append(result, convertDBTransaction(ticket)) } return result, nil } func (s *Store) GetTransactionByBranch(ctx context.Context, id int64) ([]domain.Transaction, error) { transaction, err := s.queries.GetTransactionByBranch(ctx, id) if err != nil { return nil, err } var result []domain.Transaction = make([]domain.Transaction, 0, len(transaction)) for _, ticket := range transaction { result = append(result, convertDBTransaction(ticket)) } return result, nil } func (s *Store) UpdateTransactionVerified(ctx context.Context, id int64, verified bool, approvedBy int64, approverName string) error { err := s.queries.UpdateTransactionVerified(ctx, dbgen.UpdateTransactionVerifiedParams{ ID: id, ApprovedBy: pgtype.Int8{ Int64: approvedBy, Valid: true, }, Verified: verified, ApproverName: pgtype.Text{ String: approverName, Valid: true, }, }) return err } // GetTransactionTotals returns total deposits and withdrawals func (s *Store) GetTransactionTotals(ctx context.Context, filter domain.ReportFilter) (deposits, withdrawals domain.Currency, err error) { query := `SELECT COALESCE(SUM(CASE WHEN type = 'deposit' THEN amount ELSE 0 END), 0) as deposits, COALESCE(SUM(CASE WHEN type = 'withdrawal' THEN amount ELSE 0 END), 0) as withdrawals FROM transactions` args := []interface{}{} argPos := 1 if filter.CompanyID.Valid { query += fmt.Sprintf(" WHERE company_id = $%d", argPos) args = append(args, filter.CompanyID.Value) argPos++ } else if filter.BranchID.Valid { query += fmt.Sprintf(" WHERE branch_id = $%d", argPos) args = append(args, filter.BranchID.Value) argPos++ } else if filter.UserID.Valid { query += fmt.Sprintf(" WHERE cashier_id = $%d", argPos) args = append(args, filter.UserID.Value) argPos++ } if filter.StartTime.Valid { query += fmt.Sprintf(" AND %screated_at >= $%d", func() string { if len(args) == 0 { return "" } return " " }(), argPos) args = append(args, filter.StartTime.Value) argPos++ } if filter.EndTime.Valid { query += fmt.Sprintf(" AND created_at <= $%d", argPos) args = append(args, filter.EndTime.Value) argPos++ } row := s.conn.QueryRow(ctx, query, args...) err = row.Scan(&deposits, &withdrawals) if err != nil { return 0, 0, fmt.Errorf("failed to get transaction totals: %w", err) } return deposits, withdrawals, nil } // GetBranchTransactionTotals returns transaction totals by branch func (s *Store) GetBranchTransactionTotals(ctx context.Context, filter domain.ReportFilter) (map[int64]domain.BranchTransactions, error) { query := `SELECT branch_id, COALESCE(SUM(CASE WHEN type = 'deposit' THEN amount ELSE 0 END), 0) as deposits, COALESCE(SUM(CASE WHEN type = 'withdrawal' THEN amount ELSE 0 END), 0) as withdrawals FROM transactions` args := []interface{}{} argPos := 1 if filter.CompanyID.Valid { query += fmt.Sprintf(" WHERE company_id = $%d", argPos) args = append(args, filter.CompanyID.Value) argPos++ } if filter.StartTime.Valid { query += fmt.Sprintf(" AND %screated_at >= $%d", func() string { if len(args) == 0 { return " WHERE " } return " AND " }(), argPos) args = append(args, filter.StartTime.Value) argPos++ } if filter.EndTime.Valid { query += fmt.Sprintf(" AND created_at <= $%d", argPos) args = append(args, filter.EndTime.Value) argPos++ } query += " GROUP BY branch_id" rows, err := s.conn.Query(ctx, query, args...) if err != nil { return nil, fmt.Errorf("failed to query branch transaction totals: %w", err) } defer rows.Close() totals := make(map[int64]domain.BranchTransactions) for rows.Next() { var branchID int64 var transactions domain.BranchTransactions if err := rows.Scan(&branchID, &transactions.Deposits, &transactions.Withdrawals); err != nil { return nil, fmt.Errorf("failed to scan branch transaction totals: %w", err) } totals[branchID] = transactions } if err := rows.Err(); err != nil { return nil, fmt.Errorf("rows error: %w", err) } return totals, nil }