package main

import (
	"flag"
	"fmt"
	"log/slog"
	"os"

	"focs.ji.sjtu.edu.cn/git/FOCS-dev/JOJ3/pkg/healthcheck"
)

// parseMultiValueFlag parses a multi-value command-line flag and appends its values to the provided slice.
// It registers a flag with the specified name and description, associating it with a multiStringValue receiver.
func parseMultiValueFlag(values *[]string, flagName, description string) {
	flag.Var((*multiStringValue)(values), flagName, description)
}

type multiStringValue []string

// Set appends a new value to the multiStringValue slice.
// It satisfies the flag.Value interface, allowing multiStringValue to be used as a flag value.
func (m *multiStringValue) Set(value string) error {
	*m = append(*m, value)
	return nil
}

func (m *multiStringValue) String() string {
	return fmt.Sprintf("%v", *m)
}

func setupSlog() {
	opts := &slog.HandlerOptions{}
	handler := slog.NewTextHandler(os.Stderr, opts)
	logger := slog.New(handler)
	slog.SetDefault(logger)
}

// Generally, err is used for runtime errors, and checkRes is used for the result of the checks.
func main() {
	var gitWhitelist, metaFile, releaseTags []string
	rootDir := flag.String("root", "", "")
	repo := flag.String("repo", "", "")
	droneBranch := flag.String("droneBranch", "", "")
	releaseCategories := flag.String("releaseCategories", "", "")
	releaseNumber := flag.Int("releaseNumber", 0, "")
	// FIXME: for drone usage
	adminDir := flag.String("admin", "", "") // adminDir is for config files
	parseMultiValueFlag(&gitWhitelist, "whitelist", "")
	parseMultiValueFlag(&metaFile, "meta", "")
	parseMultiValueFlag(&releaseTags, "releaseTags", "")
	flag.Parse()
	setupSlog()
	var err error
	err = healthcheck.RepoSize()
	if err != nil {
		fmt.Printf("## Repo Size Check Failed:\n%s\n", err.Error())
	}
	err = healthcheck.ForbiddenCheck(*rootDir, gitWhitelist, *repo, *droneBranch)
	if err != nil {
		fmt.Printf("## Forbidden File Check Failed:\n%s\n", err.Error())
	}
	err = healthcheck.MetaCheck(*rootDir, metaFile)
	if err != nil {
		fmt.Printf("## Forbidden File Check Failed:\n%s\n", err.Error())
	}
	err = healthcheck.NonAsciiFiles(*rootDir)
	if err != nil {
		fmt.Printf("## Non-ASCII Characters File Check Failed:\n%s\n", err.Error())
	}
	err = healthcheck.NonAsciiMsg(*rootDir)
	if err != nil {
		fmt.Printf("## Non-ASCII Characters Commit Message Check Failed:\n%s\n", err.Error())
	}
	// TODO: find a way to test the release tag
	err = healthcheck.CheckReleases(*rootDir, *releaseCategories, *releaseNumber)
	if err != nil {
		fmt.Printf("## Release Tag Check Failed:\n%s\n", err.Error())
	}
	// FIXME: for drone usage
	if adminDir != nil && *adminDir != "" {
		err = healthcheck.VerifyDirectory(*rootDir, *adminDir)
		if err != nil {
			fmt.Printf("## Directory File Check Failed:\n%s\n", err.Error())
		}
	}
}