package main

import (
	"context"
	"flag"
	"fmt"
	"os"
	"os/signal"
	"syscall"

	"github.com/reposurvey/pipeline/client"
	"github.com/reposurvey/pipeline/config"
	"github.com/reposurvey/pipeline/logger"
	"github.com/reposurvey/pipeline/tasks"
)

func main() {
	// Parse command line flags
	taskFlag := flag.String("task", "", "Task to run: repos, prs, enrich, llm_enhance, render, tokenize")
	flag.Parse()

	// Load configuration
	cfg := config.LoadConfig()

	fmt.Println("=== GitHub Data Synthesis Pipeline ===")
	fmt.Printf("Task: %s\n", *taskFlag)
	fmt.Printf("Proxy: %s\n", cfg.ProxyBaseURL)
	fmt.Printf("Max Concurrency (Online Tasks): %d\n", cfg.MaxConcurrency)
	fmt.Printf("Max Concurrency (Offline Tasks): %d\n", cfg.OfflineConcurrency)
	fmt.Printf("LLM Concurrency: %d\n", cfg.LLMConcurrency)
	fmt.Printf("Batch Size: %d\n", cfg.BatchSize)
	fmt.Printf("Data Directory: %s\n", cfg.DataDir)
	fmt.Println("======================================")

	// Create GitHub client
	githubClient := client.NewGithubClient(cfg.ProxyBaseURL, cfg.MaxIdleConns)

	// Create failure logger
	failureLogger, err := logger.NewLogger(cfg.DataDir + "/failures.jsonl")
	if err != nil {
		fmt.Printf("Failed to create failure logger: %v\n", err)
		os.Exit(1)
	}
	defer failureLogger.Close()

	// Create context with cancellation
	ctx, cancel := context.WithCancel(context.Background())
	defer cancel()

	// Handle graceful shutdown
	sigChan := make(chan os.Signal, 1)
	signal.Notify(sigChan, os.Interrupt, syscall.SIGTERM)
	go func() {
		sig := <-sigChan
		fmt.Printf("\n[INFO] Received signal %v, shutting down gracefully...\n", sig)
		cancel()
	}()

	// Run the selected task
	var taskErr error
	switch *taskFlag {
	case "repos":
		taskErr = runTaskRepos(ctx, cfg, githubClient)
	case "prs":
		taskErr = runTaskPRs(ctx, cfg, githubClient)
	case "enrich":
		taskErr = runTaskEnrich(ctx, cfg, githubClient)
	case "llm_enhance":
		taskErr = runTaskLLMEnhance(ctx, cfg)
	case "render":
		taskErr = runTaskRender(ctx, cfg, githubClient)
	case "tokenize":
		taskErr = runTaskTokenize(ctx, cfg, githubClient)
	default:
		fmt.Printf("Unknown task: %s\nAvailable tasks: repos, prs, enrich, llm_enhance, render, tokenize\n", *taskFlag)
		os.Exit(1)
	}

	if taskErr != nil {
		fmt.Printf("[ERROR] Task failed: %v\n", taskErr)
		os.Exit(1)
	}

	fmt.Println("[INFO] Pipeline completed successfully")
}

// runTaskRepos executes the repository survey task (Task 1)
func runTaskRepos(ctx context.Context, cfg *config.Config, client *client.GithubClient) error {
	task, err := tasks.NewTask1RepoSurvey(cfg, client)
	if err != nil {
		return fmt.Errorf("failed to create repos task: %w", err)
	}
	defer task.Close()

	return task.Run(ctx)
}

// runTaskPRs executes the PR ingestion task (Task 2)
func runTaskPRs(ctx context.Context, cfg *config.Config, client *client.GithubClient) error {
	task, err := tasks.NewTask2PRIngestion(cfg, client)
	if err != nil {
		return fmt.Errorf("failed to create prs task: %w", err)
	}
	defer task.Close()

	return task.Run(ctx)
}

// runTaskEnrich executes the PR enrichment task (Task 3)
func runTaskEnrich(ctx context.Context, cfg *config.Config, client *client.GithubClient) error {
	task, err := tasks.NewTask3EnrichPR(cfg, client)
	if err != nil {
		return fmt.Errorf("failed to create enrich task: %w", err)
	}
	defer task.Close()

	return task.Run(ctx)
}

// runTaskLLMEnhance executes the LLM enhancement task (Task 4)
func runTaskLLMEnhance(ctx context.Context, cfg *config.Config) error {
	task, err := tasks.NewTask4LLMEnhance(cfg)
	if err != nil {
		return fmt.Errorf("failed to create llm_enhance task: %w", err)
	}
	defer task.Close()

	return task.Run(ctx)
}

// runTaskRender executes the PR text rendering task (Task 5)
func runTaskRender(ctx context.Context, cfg *config.Config, client *client.GithubClient) error {
	task, err := tasks.NewTask5RenderText(cfg, client)
	if err != nil {
		return fmt.Errorf("failed to create render task: %w", err)
	}
	defer task.Close()

	return task.Run(ctx)
}

// runTaskTokenize executes the PR tokenization task (Task 6)
func runTaskTokenize(ctx context.Context, cfg *config.Config, client *client.GithubClient) error {
	task, err := tasks.NewTask6Tokenization(cfg, client)
	if err != nil {
		return fmt.Errorf("failed to create tokenize task: %w", err)
	}
	defer task.Close()

	return task.Run(ctx)
}
