Laravel Построитель запросов вычитает COUNT по подзапросу в запросе, сгруппированном по ID - PullRequest
3 голосов
/ 20 февраля 2020

Я застрял в этой проблеме, с которой столкнулся в запросе Laravel.

Во-первых, у меня есть этот запрос, который я использую для получения транзакций

$transactions = Transaction::select(
    'customers.member_id as member_id',
    DB::raw('
        CASE
            WHEN transactions.customer_id IS NOT NULL
                THEN
                    customers.name 
            ELSE 
                "Tanpa pelanggan" 
        END AS customer_name
    '),
    'customers.phone as customer_phone',
    DB::raw('COUNT(transactions.id) as sum_qty'),
    DB::raw('SUM(final_amount) as sum_amount')
)
->join('outlets', 'outlets.id', '=', 'transactions.outlet_id')
->leftJoin('customers', 'customers.id', '=', 'transactions.customer_id')
->where('outlets.company_id', '=', $companyID)
->where(function ($q) use ($permittedOutlets) {
    if (!empty($permittedOutlets)) {
        $q->whereIn('transactions.outlet_id', $permittedOutlets);
    }
})
->where(function ($query) use ($companyID, $outletID, $startDate, $endDate, $filterData) {
    if ($outletID != 0) {
        $query->where('transactions.outlet_id', '=', $outletID);
    }
    if (!empty($startDate) && !empty($endDate)) {
        if ($startDate == $endDate) {
            $query->whereDate(deviceTimestamp(), '=', $startDate);
        } else {
            $query->whereDate(deviceTimestamp(), '>=', $startDate)
                ->whereDate(deviceTimestamp(), '<=', $endDate);
        }
    }
    if ($filterData != "") {
        $query->where(function ($q) use ($filterData) {
            $q->where('customers.name', 'like', '%' . $filterData . '%')
                ->orWhere('customers.phone', 'like', '%' . $filterData . '%');
        });
    }
})
->isSuccessAndVoidedInDifferentDate()
->orderBy($sortField, $sort)
->groupBy('transactions.customer_id');

и для транзакций isSuccessAndVoidedInDifferentDate Объем модели

$query->whereRaw('
    CASE 
        WHEN transactions.void_id IS NOT NULL
            THEN 
                EXISTS (
                    SELECT *
                    FROM transactions as transactions2
                    WHERE transactions.void_id = transactions2.id
                    AND transactions.created_at <> transactions2.created_at
                    LIMIT 1
                )                
        WHEN transactions.status = \''.self::STATUS_VOID.'\'
            THEN 
                EXISTS (
                    SELECT *
                    FROM transactions as transactions2
                    WHERE transactions.id = transactions2.void_id
                    AND transactions.created_at <> transactions2.created_at
                    LIMIT 1
                )   
        ELSE 
                TRUE
    END
');

Для необработанных MySql Запрос

SELECT 
    `customers`.`member_id` AS `member_id`,
    CASE
        WHEN transactions.customer_id IS NOT NULL THEN customers.name
        ELSE 'Tanpa pelanggan'
    END AS customer_name,
    `customers`.`phone` AS `customer_phone`,
    COUNT(transactions.id) AS sum_qty,
    SUM(final_amount) AS sum_amount
FROM
    `transactions`
        INNER JOIN
    `outlets` ON `outlets`.`id` = `transactions`.`outlet_id`
        LEFT JOIN
    `customers` ON `customers`.`id` = `transactions`.`customer_id`
WHERE
    `outlets`.`company_id` = 153113
        AND (`transactions`.`outlet_id` IN (9164))
        AND (DATE(ADDDATE(transactions.device_timestamp,
            INTERVAL transactions.timezone HOUR)) >= '2020-02-01'
        AND DATE(ADDDATE(transactions.device_timestamp,
            INTERVAL transactions.timezone HOUR)) <= '2020-02-29')
        AND CASE
        WHEN
            transactions.void_id IS NOT NULL
        THEN
            EXISTS( SELECT 
                    *
                FROM
                    transactions AS transactions2
                WHERE
                    transactions.void_id = transactions2.id
                        AND transactions.created_at <> transactions2.created_at
                LIMIT 1)
        WHEN
            transactions.status = 'void'
        THEN
            EXISTS( SELECT 
                    *
                FROM
                    transactions AS transactions2
                WHERE
                    transactions.id = transactions2.void_id
                        AND transactions.created_at <> transactions2.created_at
                LIMIT 1)
        ELSE TRUE
    END
        AND `transactions`.`deleted_at` IS NULL
GROUP BY `transactions.customer_id`
ORDER BY `transactions.customer_id` ASC

Все транзакции имеют два разных статуса. Статус либо успешен, либо аннулирован. Типы транзакций делятся на две категории. Первая категория - это транзакция с клиентом, а вторая - без клиента. В этом случае транзакция без клиента будет выбрана как «Tanpa pelanggan». Все транзакции здесь сгруппированы по транзакциям.customer_id.

Этот вопрос сосредоточен на

   DB::raw('COUNT(transactions.id) as sum_qty'),

в выбранной части. Для более простого понимания положений компании, торговых точек и даты здесь можно игнорировать. Я пытаюсь создать условие, при котором счетчик в запросе будет возвращать количество успешных транзакций, вычтенное из числа аннулированных транзакций, сгруппированных по транзакциям.customer_id. Меня немного смущает вопрос о том, должен ли я выполнять подзапрос с такими же условиями в качестве основного запроса, который является своего рода повторяющимся, или каким-либо лучшим решением, которое я не нашел.

Пример: с данными состоит из:

1 транзакции отменены, остальные транзакции успешны

Ожидаемый результат:

member_id | customer_name | customer_phone| sum_qty | sum_ammount

NULL      | Sam           | 123xxxx       | 1       | 20720

NULL      | Tanpa Pelanggan| -            | 2       | 28490

Фактический результат:

member_id | customer_name | customer_phone| sum_qty | sum_ammount

NULL      | Sam           | 123xxxx       | 1       | 20720

NULL      | Tanpa Pelanggan| -            | 3       | 51800

Мой вопрос: как лучше решить эту проблему?

Заранее спасибо!

1 Ответ

0 голосов
/ 21 февраля 2020

Нашли решение!

DB::raw('SUM(CASE WHEN transactions.status != \'void\' THEN 1 ELSE 0 END) - SUM(CASE WHEN transactions.status = \'void\' THEN 1 ELSE 0 END) as sum_qty')

Это на самом деле использовать SUM и добавить условие CASE в функцию SUM.

...