Реалізація логування і звітів про результати імпорту товарів

Наша компанія займається розробкою, підтримкою та обслуговуванням сайтів будь-якої складності. Від простих односторінкових сайтів до масштабних кластерних систем, побудованих на мікро сервісах. Досвід розробників підтверджено сертифікатами від вендорів.

Розробка та обслуговування будь-яких видів сайтів:

Інформаційні сайти або веб-програми
Сайти візитки, landing page, корпоративні сайти, онлайн каталоги, квіз, промо-сайти, блоги, ресурси новин, інформаційні портали, форуми, агрегатори
Сайти або веб-програми електронної комерції
Інтернет-магазини, B2B-портали, маркетплейси, онлайн-обмінники, кешбек-сайти, біржі, дропшиппінг-платформи, парсери товарів
Веб-програми для управління бізнес-процесами
CRM-системи, ERP-системи, корпоративні портали, системи управління виробництвом, парсери інформації
Сайти або веб-програми електронних послуг
Дошки оголошень, онлайн-школи, онлайн-кінотеатри, конструктори сайтів, портали надання електронних послуг, відеохостинги, тематичні портали

Це лише деякі з технічних типів сайтів, з якими ми працюємо, і кожен із них може мати свої специфічні особливості та функціональність, а також бути адаптованим під конкретні потреби та цілі клієнта.

Пропоновані послуги
Показано 1 з 1 послугУсі 2065 послуг
Реалізація логування і звітів про результати імпорту товарів
Проста
від 1 робочого дня до 3 робочих днів
Часті питання

Наші компетенції:

Етапи розробки

Останні роботи

  • image_website-b2b-advance_0.png
    Розробка сайту компанії B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Розробка веб-додатків для компанії FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Розробка веб-сайту для компанії БЕЛФІНГРУП
    874
  • image_ecommerce_furnoro_435_0.webp
    Розробка інтернет магазину для компанії FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Розробка веб-додатків для компанії Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Розробка веб-сайту для компанії ФІКСПЕР
    851

Реалізація логування та звітів про результати імпорту товарів

Без логування імпорт — чорна скринька. Щось пішло не так вночі, менеджер з утра бачить розбіжності в каталозі, але зрозуміти причину неможливо. Гарне логування фіксує кожен крок, а звіт дає однозначну відповідь: що було, що змінилось, що зламалось.

Що потрібно фіксувати

На рівні запуску імпорту:

  • Час старту та завершення
  • Джерело даних, тип, URL/шлях до файлу
  • Підсумкові лічильники: створено / оновлено / пропущено / помилок
  • Статус: success / partial / failed
  • Користувач, який запустив імпорт (якщо вручну)

На рівні окремого рядка:

  • Номер рядка / SKU
  • Тип операції: create / update / skip / error
  • Поля, які змінилися (diff)
  • Повідомлення про помилку (якщо є)

Схема БД для логів

CREATE TABLE import_runs (
    id              serial PRIMARY KEY,
    source_id       int REFERENCES import_sources(id),
    status          varchar(20) DEFAULT 'pending',  -- pending | processing | success | partial | failed
    trigger         varchar(20) DEFAULT 'scheduled', -- scheduled | manual | webhook
    triggered_by    int REFERENCES users(id),
    file_name       varchar(500),
    file_size       bigint,
    total_rows      int DEFAULT 0,
    created_count   int DEFAULT 0,
    updated_count   int DEFAULT 0,
    skipped_count   int DEFAULT 0,
    errors_count    int DEFAULT 0,
    started_at      timestamptz,
    completed_at    timestamptz,
    duration_ms     int,
    error_message   text,
    created_at      timestamptz DEFAULT now()
);

CREATE TABLE import_row_logs (
    id          bigserial PRIMARY KEY,
    import_id   int REFERENCES import_runs(id) ON DELETE CASCADE,
    line_number int,
    sku         varchar(100),
    operation   varchar(10),  -- create | update | skip | error
    changed_fields jsonb,     -- {"price": {"old": 100, "new": 120}}
    error_code  varchar(50),
    error_msg   text,
    created_at  timestamptz DEFAULT now()
);

CREATE INDEX import_row_logs_import_id_idx ON import_row_logs (import_id);
CREATE INDEX import_row_logs_sku_idx       ON import_row_logs (sku);

Логер імпорту

class ImportLogger
{
    private ImportRun $run;
    private array     $rowBuffer = [];
    private int       $bufferSize = 500;

    public function start(int $sourceId, string $trigger, ?int $userId): void
    {
        $this->run = ImportRun::create([
            'source_id'    => $sourceId,
            'status'       => 'processing',
            'trigger'      => $trigger,
            'triggered_by' => $userId,
            'started_at'   => now(),
        ]);
    }

    public function logRow(
        int    $line,
        string $sku,
        string $operation,
        array  $changedFields = [],
        ?string $errorMsg = null,
        ?string $errorCode = null
    ): void {
        $this->rowBuffer[] = [
            'import_id'     => $this->run->id,
            'line_number'   => $line,
            'sku'           => $sku,
            'operation'     => $operation,
            'changed_fields'=> $changedFields ? json_encode($changedFields) : null,
            'error_code'    => $errorCode,
            'error_msg'     => $errorMsg,
            'created_at'    => now()->toDateTimeString(),
        ];

        if (count($this->rowBuffer) >= $this->bufferSize) {
            $this->flush();
        }
    }

    public function finish(string $status, ?string $errorMessage = null): void
    {
        $this->flush();

        $counts = DB::table('import_row_logs')
            ->where('import_id', $this->run->id)
            ->selectRaw("
                SUM(CASE WHEN operation = 'create'  THEN 1 ELSE 0 END) AS created,
                SUM(CASE WHEN operation = 'update'  THEN 1 ELSE 0 END) AS updated,
                SUM(CASE WHEN operation = 'skip'    THEN 1 ELSE 0 END) AS skipped,
                SUM(CASE WHEN operation = 'error'   THEN 1 ELSE 0 END) AS errors
            ")
            ->first();

        $this->run->update([
            'status'        => $status,
            'created_count' => $counts->created,
            'updated_count' => $counts->updated,
            'skipped_count' => $counts->skipped,
            'errors_count'  => $counts->errors,
            'completed_at'  => now(),
            'duration_ms'   => now()->diffInMilliseconds($this->run->started_at),
            'error_message' => $errorMessage,
        ]);
    }

    private function flush(): void
    {
        if (!empty($this->rowBuffer)) {
            DB::table('import_row_logs')->insert($this->rowBuffer);
            $this->rowBuffer = [];
        }
    }
}

Буферизація записів по 500 штук — замість INSERT на кожен рядок.

Фіксація diff змінених полів

private function buildDiff(Product $existing, array $newData): array
{
    $trackFields = ['price', 'qty', 'name', 'description'];
    $diff        = [];

    foreach ($trackFields as $field) {
        $old = $existing->{$field};
        $new = $newData[$field] ?? null;

        if ((string) $old !== (string) $new) {
            $diff[$field] = ['old' => $old, 'new' => $new];
        }
    }

    return $diff;
}

Агрегований звіт

class ImportReportBuilder
{
    public function build(ImportRun $run): ImportReport
    {
        $topErrors = DB::table('import_row_logs')
            ->where('import_id', $run->id)
            ->where('operation', 'error')
            ->select('error_code', DB::raw('COUNT(*) as count'), DB::raw('MIN(sku) as example_sku'))
            ->groupBy('error_code')
            ->orderByDesc('count')
            ->limit(10)
            ->get();

        $priceChanges = DB::table('import_row_logs')
            ->where('import_id', $run->id)
            ->where('operation', 'update')
            ->whereRaw("changed_fields ? 'price'")
            ->count();

        return new ImportReport(
            run: $run,
            topErrors: $topErrors,
            priceChangesCount: $priceChanges,
        );
    }
}

Повідомлення за результатами

class ImportCompletedNotification extends Notification
{
    public function toMail(mixed $notifiable): MailMessage
    {
        $run = $this->run;
        return (new MailMessage)
            ->subject("Імпорт #{$run->id}: {$run->status}")
            ->line("Джерело: {$run->source->name}")
            ->line("Створено: {$run->created_count}, оновлено: {$run->updated_count}, помилок: {$run->errors_count}")
            ->line("Тривалість: " . round($run->duration_ms / 1000, 1) . " сек")
            ->when($run->errors_count > 0, fn($m) => $m->action('Переглянути помилки', $this->reportUrl()));
    }
}

Повідомлення надсилається тільки якщо статус не success або кількість помилок перевищує поріг.

Ротація логів

Построчні логи зростають швидко. Політика зберігання:

// Artisan-команда в scheduler
$schedule->command('import:cleanup-logs --older-than=30')->weekly();
class CleanupImportLogsCommand extends Command
{
    public function handle(): void
    {
        $cutoff = now()->subDays($this->option('older-than'));

        // Видаляємо построчні логи старих успішних імпортів
        ImportRowLog::whereHas('run', fn($q) =>
            $q->where('status', 'success')->where('completed_at', '<', $cutoff)
        )->delete();

        // Зведені записи import_runs залишаємо назавжди (вони маленькі)
    }
}

Терміни реалізації

  • ImportLogger з буферизацією, таблиці в БД, фінальні лічильники — 1 день
  • Diff полів, агрегований звіт, повідомлення — +0.5 дня
  • UI перегляду логів в админці, ротація старих записів — +0.5 дня