package main

import (
	"fmt"
	"os"
	"reflect"
	"runtime"
	"time"

	_ "image/gif"
	_ "image/jpeg"
	_ "image/png"

	"github.com/aerogo/log"
	"github.com/animenotifier/arn"
	"github.com/fatih/color"
)

const (
	webPQuality = 80
)

var avatarSources []AvatarSource
var avatarOutputs []AvatarOutput
var avatarLog = log.New()

// Main
func main() {
	color.Yellow("Generating user avatars")

	// Switch to main directory
	os.Chdir("../../")

	// Log
	avatarLog.AddOutput(log.File("logs/avatar.log"))
	defer avatarLog.Flush()

	// Define the avatar sources
	avatarSources = []AvatarSource{
		&Gravatar{
			Rating:         "pg",
			RequestLimiter: time.NewTicker(250 * time.Millisecond),
		},
		&MyAnimeList{
			RequestLimiter: time.NewTicker(250 * time.Millisecond),
		},
	}

	// Define the avatar outputs
	avatarOutputs = []AvatarOutput{
		// Original - Large
		&AvatarOriginalFileOutput{
			Directory: "images/avatars/large/original/",
			Size:      arn.AvatarMaxSize,
		},

		// Original - Small
		&AvatarOriginalFileOutput{
			Directory: "images/avatars/small/original/",
			Size:      arn.AvatarSmallSize,
		},

		// WebP - Large
		&AvatarWebPFileOutput{
			Directory: "images/avatars/large/webp/",
			Size:      arn.AvatarMaxSize,
			Quality:   webPQuality,
		},

		// WebP - Small
		&AvatarWebPFileOutput{
			Directory: "images/avatars/small/webp/",
			Size:      arn.AvatarSmallSize,
			Quality:   webPQuality,
		},
	}

	if InvokeShellArgs() {
		return
	}

	// Stream of all users
	users, _ := arn.FilterUsers(func(user *arn.User) bool {
		return true
	})

	// Log user count
	println(len(users), "users")

	// Worker queue
	usersQueue := make(chan *arn.User)
	StartWorkers(usersQueue, Work)

	// We'll send each user to one of the worker threads
	for _, user := range users {
		usersQueue <- user
	}

	color.Green("Finished.")
}

// StartWorkers creates multiple workers to handle a user each.
func StartWorkers(queue chan *arn.User, work func(*arn.User)) {
	for w := 0; w < runtime.NumCPU(); w++ {
		go func() {
			for user := range queue {
				work(user)
			}
		}()
	}
}

// Work handles a single user.
func Work(user *arn.User) {
	user.Avatar = ""

	for _, source := range avatarSources {
		avatar := source.GetAvatar(user)

		if avatar == nil {
			// fmt.Println(color.RedString("✘"), reflect.TypeOf(source).Elem().Name(), user.Nick)
			continue
		}

		for _, writer := range avatarOutputs {
			err := writer.SaveAvatar(avatar)

			if err != nil {
				color.Red(err.Error())
			}
		}

		fmt.Println(color.GreenString("✔"), reflect.TypeOf(source).Elem().Name(), "|", user.Nick, "|", avatar)
		user.Avatar = "/+" + user.Nick + "/avatar"
		break
	}

	// Save avatar data
	user.Save()
}