package backend

import (
	"bufio"
	"fmt"
	"image"
	"io"
	"io/fs"
	"os"
	"path/filepath"
	"strings"

	"github.com/chai2010/webp"
	"github.com/disintegration/imaging"
	"github.com/google/uuid"
)

var ErrUnsupportedFormat error = image.ErrFormat // used internally by imaging

func checkImageUsage(c *Config, db *DB, name string) (bool, error) {
	imageWasFound := false

	if err := filepath.Walk(c.ArticleDir, func(path string, info fs.FileInfo, err error) error {
		if err != nil {
			return fmt.Errorf("error walking articles filepath: %v", err)
		}

		if !info.IsDir() {
			mdFile, err := os.Open(path)
			if err != nil {
				return fmt.Errorf("error opening article %v: %v", info.Name(), err)
			}
			defer mdFile.Close()

			scanner := bufio.NewScanner(mdFile)
			for scanner.Scan() {
				if strings.Contains(scanner.Text(), name) {
					imageWasFound = true
					return nil
				}
			}

			return scanner.Err()
		}

		return nil
	}); err != nil {
		return false, fmt.Errorf("error walking articles filepath: %v", err)
	}

	if !imageWasFound {
		users, err := db.GetAllUsers(c)
		if err != nil {
			return false, fmt.Errorf("error getting all users: %v", err)
		}

		for _, user := range users {
			if name == user.ProfilePicLink {
				imageWasFound = true
			}
		}
	}

	if !imageWasFound {
		articles, err := db.GetAllArticles()
		if err != nil {
			return false, fmt.Errorf("error getting all articles: %v", err)
		}

		for _, article := range articles {
			if name == article.BannerLink {
				imageWasFound = true
			}
		}
	}

	return imageWasFound, nil
}

func SaveImage(src io.Reader, maxHeight, maxWidth int, path string) (string, error) {
	img, err := imaging.Decode(src, imaging.AutoOrientation(true))
	if err != nil {
		if err == ErrUnsupportedFormat {
			return "", ErrUnsupportedFormat
		}
		return "", fmt.Errorf("error decoding image: %v", err)
	}

	if img.Bounds().Dy() > maxHeight {
		img = imaging.Resize(img, 0, maxHeight, imaging.Lanczos)
	}
	if img.Bounds().Dx() > maxWidth {
		img = imaging.Resize(img, maxWidth, 0, imaging.Lanczos)
	}

	filename := fmt.Sprint(uuid.New(), ".webp")
	file, err := os.Create(filepath.Join(path, filename))
	if err != nil {
		return "", fmt.Errorf("error creating new image file: %v", err)
	}
	defer file.Close()

	if err = webp.Encode(file, img, &webp.Options{Quality: 80}); err != nil {
		return "", fmt.Errorf("error encoding image as webp: %v", err)
	}

	return filename, nil
}

func CleanUpImages(c *Config, db *DB) error {
	if err := filepath.Walk(c.ImgDir, func(path string, info fs.FileInfo, err error) error {
		if err != nil {
			return fmt.Errorf("error walking images filepath: %v", err)
		}

		if !info.IsDir() {
			imageName := info.Name()
			imagePath := path

			imageWasFound, err := checkImageUsage(c, db, imageName)
			if err != nil {
				return fmt.Errorf("error checking image usage: %v", err)
			}

			if !imageWasFound {
				if err = os.Remove(imagePath); err != nil {
					return fmt.Errorf("error removing unused image: %v", err)
				}
			}
		}

		return nil
	}); err != nil {
		return fmt.Errorf("error cleaning up: %v", err)
	}

	return nil
}