Files
GenAudioBookInfo/internal/infrastructure/cover_downloader.go
Dmitriy Fofanov 402ce7f4f1 Функция: реализованы консольный логгер и презентер для обработки аудиокниг
- Добавлен ConsoleLogger для подробного логирования этапов обработки аудиокниг в консоли.

- Введен ConsolePresenter для форматированного вывода результатов сканирования в консоль.

- Создан ProcessAudioBooksUseCase для обработки полного конвейера обработки аудиокниг, включая сканирование папок, извлечение метаданных, поиск торрентов и запись результатов.

- Реализована проверка LLM для улучшения метаданных.

- Добавлена ​​обработка ошибок и логирование на всех этапах обработки.
2026-02-20 00:35:43 +03:00

73 lines
2.1 KiB
Go

// Package infrastructure реализует порт CoverDownloader —
// скачивание обложки аудиокниги по URL.
package infrastructure
import (
"context"
"fmt"
"io"
"net/http"
"os"
"path/filepath"
"time"
)
// HTTPCoverDownloader реализует domain.CoverDownloader.
type HTTPCoverDownloader struct {
httpClient *http.Client
}
// NewHTTPCoverDownloader создаёт новый экземпляр.
func NewHTTPCoverDownloader() *HTTPCoverDownloader {
return &HTTPCoverDownloader{
httpClient: &http.Client{
Timeout: 30 * time.Second,
},
}
}
// Download скачивает изображение по url и сохраняет как cover.jpg в destFolder.
// Если cover.jpg уже существует — ничего не делает.
func (d *HTTPCoverDownloader) Download(ctx context.Context, url string, destFolder string) error {
if url == "" {
return nil
}
// Проверяем, нет ли уже обложки
coverPatterns := []string{"cover.jpg", "cover.jpeg", "cover.png", "folder.jpg", "folder.png"}
for _, pattern := range coverPatterns {
if _, err := os.Stat(filepath.Join(destFolder, pattern)); err == nil {
return nil // обложка уже есть
}
}
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, nil)
if err != nil {
return fmt.Errorf("ошибка создания запроса обложки: %w", err)
}
req.Header.Set("User-Agent", "GenAudioBookInfo/1.0")
resp, err := d.httpClient.Do(req)
if err != nil {
return fmt.Errorf("ошибка скачивания обложки: %w", err)
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
return fmt.Errorf("сервер обложки вернул статус %d", resp.StatusCode)
}
coverPath := filepath.Join(destFolder, "cover.jpg")
out, err := os.Create(coverPath)
if err != nil {
return fmt.Errorf("не удалось создать файл обложки: %w", err)
}
defer out.Close()
if _, err := io.Copy(out, resp.Body); err != nil {
return fmt.Errorf("ошибка записи обложки: %w", err)
}
return nil
}