package tasks

import (
	"context"
	"errors"
	"fmt"
	"math"
	"sort"
	"sync"
	"sync/atomic"
	"time"

	"github.com/reposurvey/pipeline/client"
	"github.com/reposurvey/pipeline/config"
	"github.com/reposurvey/pipeline/models"
	"github.com/reposurvey/pipeline/parquet"
	"github.com/reposurvey/pipeline/tokenizer"
)

// prKey is a composite key for identifying PRs
type prKey struct {
	RepoID   int64
	PRNumber int64
}

// tokenizationJob represents a batch of LLM enhanced PRs to tokenize
type tokenizationJob struct {
	prs []models.LLMEnhancedPRData
}

// tokenizationThroughputMonitor tracks and reports tokenization pipeline throughput
type tokenizationThroughputMonitor struct {
	mu           sync.Mutex
	prsProcessed int64
	prsTokenized int64
	prsDiscarded int64
	tokensCount  int64
	startTime    time.Time
	writer       *parquet.ParallelBatchWriter[models.TokenizedPRData]
}

func newTokenizationThroughputMonitor(writer *parquet.ParallelBatchWriter[models.TokenizedPRData]) *tokenizationThroughputMonitor {
	return &tokenizationThroughputMonitor{
		startTime: time.Now(),
		writer:    writer,
	}
}

func (m *tokenizationThroughputMonitor) addStats(processed, tokenized, discarded int, tokens int64) {
	m.mu.Lock()
	defer m.mu.Unlock()
	m.prsProcessed += int64(processed)
	m.prsTokenized += int64(tokenized)
	m.prsDiscarded += int64(discarded)
	m.tokensCount += tokens
}

func (m *tokenizationThroughputMonitor) report() {
	m.mu.Lock()
	defer m.mu.Unlock()

	elapsed := time.Since(m.startTime).Seconds()
	processRate := float64(m.prsProcessed) / elapsed
	tokenRate := float64(m.tokensCount) / elapsed

	// Get data written size
	totalBytes := m.writer.GetTotalSize()
	totalGB := float64(totalBytes) / (1024 * 1024 * 1024)
	mbps := float64(totalBytes) / (1024 * 1024) / elapsed

	fmt.Printf("[THROUGHPUT] PRs: %d (%.1f/s) | Tokenized: %d | Discarded: %d | Tokens: %d (%.1f/s) | Data: %.2f GB (%.2f MB/s) | Elapsed: %.1fs\n",
		m.prsProcessed, processRate, m.prsTokenized, m.prsDiscarded, m.tokensCount, tokenRate, totalGB, mbps, elapsed)
}

// Task6Tokenization implements PR tokenization and dataset preparation using a single Rust worker
// Uses async message queue pattern: producers send requests, harvester receives responses
type Task6Tokenization struct {
	cfg         *config.Config
	writer      *parquet.ParallelBatchWriter[models.TokenizedPRData]
	statsWriter *parquet.ParallelBatchWriter[models.TokenStatistics]

	// Channel for piping jobs from reader to producer workers
	jobChan chan tokenizationJob

	// Throughput monitoring
	monitor *tokenizationThroughputMonitor

	// Filtered PR IDs from Task 2
	filteredPRs map[prKey]struct{}

	// Deduplication: track seen (RepoID, PRID) pairs
	seenPRs map[prKey]struct{}

	// Single Rust tokenizer worker with internal thread pool
	rustWorker *tokenizer.RustTokenizerClient

	// Statistics tracking (Stage 1 filters from loadFilteredPRs)
	mu                      sync.Mutex
	stage1BotFiltered       int64
	stage1BenchmarkFiltered int64

	// Stage 2+ filters (from sendJobBatch and later)
	totalFiltered        int64
	issueCommentFiltered int64
	duplicateFiltered    int64

	// Token length statistics
	tokenLengths []int32 // All token lengths for statistics

	// Async pattern: track pending requests
	pendingRequests atomic.Int64
}

// NewTask6Tokenization creates a new PR tokenization task
func NewTask6Tokenization(cfg *config.Config, client *client.GithubClient) (*Task6Tokenization, error) {
	// Create tokenized dataset writer
	writer, writerErr := parquet.NewParallelBatchWriter[models.TokenizedPRData](
		cfg.TokenizedDatasetDir,
		cfg.PRBatchSize,
		cfg.MaxFileSize,
		cfg.FlushInterval,
		32,
	)
	if writerErr != nil {
		return nil, fmt.Errorf("failed to create tokenized dataset writer: %w", writerErr)
	}

	// Create statistics writer
	statsWriter, statsErr := parquet.NewParallelBatchWriter[models.TokenStatistics](
		cfg.TokenStatsDir,
		cfg.PRBatchSize,
		cfg.MaxFileSize,
		cfg.FlushInterval,
		32,
	)
	if statsErr != nil {
		return nil, fmt.Errorf("failed to create statistics writer: %w", statsErr)
	}

	// Initialize single Rust tokenizer worker with internal thread pool
	// This single process will handle all tokenization with 128 workers in groups of 16
	fmt.Println("[INFO] Starting Rust tokenizer worker with internal thread pool")

	rustConfig := tokenizer.RustWorkerConfig{
		WorkerPath:          cfg.RustTokenizerPath,
		Model:               cfg.TokenizerModel,
		Workers:             cfg.RustWorkers,
		ThreadsPerTokenizer: cfg.RustThreadsPerTokenizer,
		MaxMessageSize:      cfg.MaxMessageSize,
	}

	rustWorker, err := tokenizer.NewRustTokenizerClient(rustConfig)
	if err != nil {
		return nil, fmt.Errorf("failed to start Rust worker: %w", err)
	}

	return &Task6Tokenization{
		cfg:         cfg,
		writer:      writer,
		statsWriter: statsWriter,
		jobChan:     make(chan tokenizationJob, cfg.OfflineConcurrency),
		monitor:     newTokenizationThroughputMonitor(writer),
		filteredPRs: make(map[prKey]struct{}),
		seenPRs:     make(map[prKey]struct{}),
		rustWorker:  rustWorker,
	}, nil
}

// Run starts the PR tokenization process with async message queue pattern
func (t *Task6Tokenization) Run(ctx context.Context) error {
	fmt.Println("[INFO] Starting Task 6: Tokenization and Dataset Preparation (Async Mode)")

	// Stage 1: Load filtered PR IDs from Task 2
	if err := t.loadFilteredPRs(ctx); err != nil {
		return fmt.Errorf("failed to load filtered PRs: %w", err)
	}

	// Create batch reader for LLM Enhanced PR data (from Task 4)
	reader, err := parquet.NewParallelBatchReader[models.LLMEnhancedPRData](t.cfg.LLMEnhancedPRsDir, t.cfg.PRBatchSize, t.cfg.OfflineConcurrency, 2)
	if err != nil {
		return fmt.Errorf("failed to create LLM enhanced PR reader: %w", err)
	}

	fmt.Printf("[INFO] Found %d LLM enhanced PR parquet files to process\n", reader.GetFileCount())

	// Start throughput monitor
	monitorCtx, monitorCancel := context.WithCancel(ctx)
	defer monitorCancel()
	go t.runThroughputMonitor(monitorCtx)

	// Start harvester goroutine (receives responses from Rust worker)
	harvesterDone := make(chan error, 1)
	go func() {
		harvesterDone <- t.runHarvester(ctx)
	}()

	// Start producer worker pool (sends requests to Rust worker)
	producersDone := make(chan error, 1)
	go func() {
		if err := t.runProducers(ctx); err != nil {
			producersDone <- fmt.Errorf("producers error: %w", err)
		}
		close(producersDone)
	}()

	// Read and dispatch PR batches to producers
	for {
		select {
		case <-ctx.Done():
			close(t.jobChan)
			return ctx.Err()
		default:
		}

		batch, hasMore, err := reader.ReadBatch()
		if err != nil {
			close(t.jobChan)
			return fmt.Errorf("failed to read enriched PR batch: %w", err)
		}

		if len(batch) > 0 {
			job := tokenizationJob{
				prs: batch,
			}

			select {
			case t.jobChan <- job:
			case <-ctx.Done():
				close(t.jobChan)
				return ctx.Err()
			}
		}

		if !hasMore {
			break
		}
	}

	// Signal producers that no more jobs will be sent
	close(t.jobChan)

	// Wait for producers to complete sending all requests
	if err := <-producersDone; err != nil {
		return err
	}

	fmt.Println("[INFO] All requests sent, closing stdin to signal Rust worker...")

	// Close stdin to signal Rust worker that no more requests will be sent
	// This allows the Rust reader thread to exit and workers to finish processing
	if err := t.rustWorker.CloseStdin(); err != nil {
		fmt.Printf("[WARN] Error closing stdin: %v\n", err)
	}

	fmt.Println("[INFO] Waiting for responses...")

	// Wait for all responses to be processed
	for t.pendingRequests.Load() > 0 {
		time.Sleep(100 * time.Millisecond)
	}

	// Close Rust worker (this will wait for the process to exit)
	if err := t.rustWorker.Close(); err != nil {
		fmt.Printf("[WARN] Error closing Rust worker: %v\n", err)
	}

	// Wait for harvester to finish
	if err := <-harvesterDone; err != nil {
		return err
	}

	// Final report
	t.monitor.report()
	t.printFilteringStats()
	t.printTokenLengthStatistics()
	fmt.Println("[INFO] Task 6 completed successfully")
	return nil
}

// loadFilteredPRs reads Task 2 output and builds a set of valid PR IDs
// Returns counts for each filter applied in Stage 1
func (t *Task6Tokenization) loadFilteredPRs(ctx context.Context) error {
	fmt.Println("[INFO] Stage 1: Loading and filtering PRs from Task 2...")
	startTime := time.Now()

	reader, err := parquet.NewParallelBatchReader[models.PRMetadata](t.cfg.RawPRsDir, 50000, t.cfg.OfflineConcurrency, 2)
	if err != nil {
		return err
	}

	count := 0
	botFiltered := 0
	benchmarkFiltered := 0
	passed := 0

	for {
		select {
		case <-ctx.Done():
			return ctx.Err()
		default:
		}

		batch, hasMore, err := reader.ReadBatch()
		if err != nil {
			return err
		}

		for _, pr := range batch {
			// Apply filters on single-row metadata
			// Filter 1: Exclude bot authors
			if pr.AuthorType == "Bot" {
				botFiltered++
				continue
			}

			// Filter 2: Exclude SWE-Bench benchmark repos
			if config.SWEBenchRepos[pr.RepoName] {
				benchmarkFiltered++
				continue
			}

			// Use composite key (RepoID, PRNumber)
			key := prKey{
				RepoID:   pr.RepoID,
				PRNumber: pr.PRNumber,
			}
			t.filteredPRs[key] = struct{}{}
			passed++
		}
		count += len(batch)

		if !hasMore {
			break
		}
	}

	// Store Stage 1 filter counts
	t.stage1BotFiltered = int64(botFiltered)
	t.stage1BenchmarkFiltered = int64(benchmarkFiltered)

	fmt.Printf("[INFO] Stage 1 complete: %d total PRs, %d passed filters (Bot: %d, Benchmark: %d) in %.1fs\n",
		count, passed, botFiltered, benchmarkFiltered, time.Since(startTime).Seconds())
	return nil
}

// runThroughputMonitor reports throughput statistics every 30 seconds
func (t *Task6Tokenization) runThroughputMonitor(ctx context.Context) {
	ticker := time.NewTicker(30 * time.Second)
	defer ticker.Stop()

	for {
		select {
		case <-ctx.Done():
			t.monitor.report()
			return
		case <-ticker.C:
			t.monitor.report()
		}
	}
}

// runProducers implements the producer worker pool that sends requests to Rust worker
func (t *Task6Tokenization) runProducers(ctx context.Context) error {
	workerCount := t.cfg.OfflineConcurrency
	fmt.Printf("[INFO] Starting %d producer workers for sending tokenization requests\n", workerCount)

	errChan := make(chan error, workerCount)

	for i := 0; i < workerCount; i++ {
		go func(workerID int) {
			for job := range t.jobChan {
				if err := t.sendJobBatch(ctx, job); err != nil {
					if !errors.Is(err, context.Canceled) {
						fmt.Printf("[ERROR] Producer %d failed to send batch: %v\n", workerID, err)
					}
				}
			}
			errChan <- nil
		}(i)
	}

	// Wait for all producers to complete
	for i := 0; i < workerCount; i++ {
		if err := <-errChan; err != nil {
			return err
		}
	}

	return nil
}

// runHarvester receives responses from Rust worker and writes to parquet
func (t *Task6Tokenization) runHarvester(ctx context.Context) error {
	fmt.Println("[INFO] Starting harvester for receiving tokenization responses")

	for {
		select {
		case <-ctx.Done():
			return ctx.Err()
		default:
		}

		// Receive response from Rust worker (blocking)
		resp, err := t.rustWorker.ReceiveResponse()
		if err != nil {
			// Channel closed, normal shutdown
			if err.Error() == "response channel closed" {
				fmt.Println("[INFO] Harvester: response channel closed, shutting down")
				return nil
			}
			return fmt.Errorf("harvester failed to receive response: %w", err)
		}

		// Process response
		if err := t.processResponse(ctx, resp); err != nil {
			if !errors.Is(err, context.Canceled) {
				fmt.Printf("[ERROR] Harvester failed to process response: %v\n", err)
			}
		}

		// Decrement pending requests counter by number of results
		t.pendingRequests.Add(-int64(len(resp.Results)))
	}
}

// sendJobBatch filters PRs, renders text, and sends tokenization request to Rust worker (non-blocking)
func (t *Task6Tokenization) sendJobBatch(ctx context.Context, job tokenizationJob) error {
	// Filter PRs and render text
	var prTexts []tokenizer.PRText
	issueCommentFiltered := 0
	duplicateFiltered := 0

	for _, pr := range job.prs {
		select {
		case <-ctx.Done():
			return ctx.Err()
		default:
		}

		// 1. Check if PR is in filtered set (already filtered by Stage 1: bots and benchmarks)
		key := prKey{
			RepoID:   pr.RepoID,
			PRNumber: pr.PRID,
		}
		if _, ok := t.filteredPRs[key]; !ok {
			// Already filtered in Stage 1, skip silently
			continue
		}

		// 2. Deduplication: skip if we've already seen this (RepoID, PRID) pair
		t.mu.Lock()
		if _, seen := t.seenPRs[key]; seen {
			t.mu.Unlock()
			duplicateFiltered++
			continue
		}
		t.seenPRs[key] = struct{}{}
		t.mu.Unlock()

		// 3. Additional filters: issue comment count
		if pr.Issue != nil && len(pr.Issue.Comments) > 20 {
			issueCommentFiltered++
			continue
		}

		// 4. Render text using exported function from Task 5 (LLM enhanced version)
		text, err := RenderWithEdits(pr)
		if err != nil {
			fmt.Printf("[WARN] Failed to render PR %s#%d: %v\n", pr.RepoName, pr.PRID, err)
			continue
		}

		prTexts = append(prTexts, tokenizer.PRText{
			RepoID:   pr.RepoID,
			RepoName: pr.RepoName,
			PRID:     pr.PRID,
			Text:     text,
		})
	}

	// Update filtering stats (Stage 2+ filters only)
	t.mu.Lock()
	t.issueCommentFiltered += int64(issueCommentFiltered)
	t.duplicateFiltered += int64(duplicateFiltered)
	t.mu.Unlock()

	if len(prTexts) == 0 {
		return nil
	}

	// 5. Send PRs to Rust worker for tokenization
	// Try sending all PRs together first, if that fails due to size, send individually
	t.pendingRequests.Add(int64(len(prTexts)))

	err := t.rustWorker.SendRequest(prTexts, int32(t.cfg.MaxTokens))
	if err != nil {
		// If batch is too large, try sending PRs individually
		if len(prTexts) > 1 {
			fmt.Printf("[WARN] Batch too large (%d PRs), sending individually: %v\n", len(prTexts), err)
			t.pendingRequests.Add(-int64(len(prTexts))) // Reset counter

			successCount := 0
			for _, prText := range prTexts {
				if err := t.rustWorker.SendRequest([]tokenizer.PRText{prText}, int32(t.cfg.MaxTokens)); err != nil {
					fmt.Printf("[WARN] Skipping large PR %s#%d (size error): %v\n", prText.RepoName, prText.PRID, err)
					// Write a discarded result for this PR
					t.writeDiscardedResult(ctx, prText, "message_too_large")
				} else {
					successCount++
				}
			}
			t.pendingRequests.Add(int64(successCount))
		} else {
			// Single PR is too large, skip it
			t.pendingRequests.Add(-1)
			fmt.Printf("[WARN] Skipping extremely large PR %s#%d: %v\n", prTexts[0].RepoName, prTexts[0].PRID, err)
			t.writeDiscardedResult(ctx, prTexts[0], "message_too_large")
		}
	}

	return nil
}

// writeDiscardedResult writes a discarded result for a PR that couldn't be tokenized
func (t *Task6Tokenization) writeDiscardedResult(ctx context.Context, prText tokenizer.PRText, reason string) {
	// Write statistics for discarded PR
	stats := models.TokenStatistics{
		RepoID:     prText.RepoID,
		RepoName:   prText.RepoName,
		PRID:       prText.PRID,
		TokenCount: 0,
		ByteSize:   int32(len(prText.Text)),
		Discarded:  true,
	}

	if err := t.statsWriter.Write(stats); err != nil {
		if !errors.Is(err, context.Canceled) {
			fmt.Printf("[ERROR] Failed to write discard stats for PR %s#%d: %v\n", prText.RepoName, prText.PRID, err)
		}
	}

	// Update monitor stats
	t.monitor.addStats(1, 0, 1, 0)
}

// processResponse processes a tokenization response from Rust worker
func (t *Task6Tokenization) processResponse(ctx context.Context, resp *tokenizer.TokenizeResponse) error {
	if resp.Status != "success" {
		errMsg := "unknown error"
		if resp.Error != nil {
			errMsg = *resp.Error
		}
		return fmt.Errorf("tokenization failed: %s", errMsg)
	}

	tokenized := 0
	discarded := 0
	var totalTokens int64 = 0

	for _, result := range resp.Results {
		select {
		case <-ctx.Done():
			return ctx.Err()
		default:
		}

		// Write statistics
		stats := models.TokenStatistics{
			RepoID:     result.RepoID,
			RepoName:   result.RepoName,
			PRID:       result.PRID,
			TokenCount: result.TokenCount,
			ByteSize:   result.ByteSize,
			Discarded:  result.Discarded,
		}

		if err := t.statsWriter.Write(stats); err != nil {
			if !errors.Is(err, context.Canceled) {
				fmt.Printf("[ERROR] Failed to write stats for PR %s#%d: %v\n", result.RepoName, result.PRID, err)
			}
		}

		// Write tokenized data if not discarded
		if !result.Discarded && result.TokenIDs != nil {
			data := models.TokenizedPRData{
				RepoID:     result.RepoID,
				RepoName:   result.RepoName,
				PRID:       result.PRID,
				TokenIDs:   result.TokenIDs,
				TokenCount: result.TokenCount,
				ByteSize:   result.ByteSize,
			}

			if err := t.writer.Write(data); err != nil {
				if !errors.Is(err, context.Canceled) {
					fmt.Printf("[ERROR] Failed to write tokenized data for PR %s#%d: %v\n", result.RepoName, result.PRID, err)
				}
				continue
			}
			tokenized++
			totalTokens += int64(result.TokenCount)
		} else {
			discarded++
		}
	}

	processed := len(resp.Results)
	t.monitor.addStats(processed, tokenized, discarded, totalTokens)

	// Collect token lengths for statistics
	t.mu.Lock()
	t.totalFiltered += int64(processed - tokenized - discarded)
	for _, result := range resp.Results {
		t.tokenLengths = append(t.tokenLengths, result.TokenCount)
	}
	t.mu.Unlock()

	return nil
}

// printFilteringStats prints filtering statistics
func (t *Task6Tokenization) printFilteringStats() {
	t.mu.Lock()
	defer t.mu.Unlock()

	fmt.Println("\n[FILTERING STATS]")
	fmt.Println("Stage 1 (loadFilteredPRs):")
	fmt.Printf("  Bot authors filtered: %d\n", t.stage1BotFiltered)
	fmt.Printf("  Benchmark repos filtered: %d\n", t.stage1BenchmarkFiltered)
	fmt.Println("Stage 2+ (sendJobBatch and later):")
	fmt.Printf("  Duplicate PRs filtered: %d\n", t.duplicateFiltered)
	fmt.Printf("  Issue comments > 20 filtered: %d\n", t.issueCommentFiltered)
	fmt.Printf("  Token length filtered: %d\n", t.totalFiltered)
}

// printTokenLengthStatistics prints comprehensive token length statistics
func (t *Task6Tokenization) printTokenLengthStatistics() {
	t.mu.Lock()
	lengths := make([]int32, len(t.tokenLengths))
	copy(lengths, t.tokenLengths)
	t.mu.Unlock()

	if len(lengths) == 0 {
		fmt.Println("\n[TOKEN LENGTH STATISTICS]")
		fmt.Println("  No tokenized data available")
		return
	}

	// Sort for percentile calculations
	sort.Slice(lengths, func(i, j int) bool {
		return lengths[i] < lengths[j]
	})

	// Print statistics for all data (with retention analysis)
	t.printStatsForDataset("ALL DATA", lengths, true)

	// Print statistics for filtered data (using MaxTokens from config)
	maxTokens := int32(t.cfg.MaxTokens)
	var filteredLengths []int32
	for _, l := range lengths {
		if l <= maxTokens {
			filteredLengths = append(filteredLengths, l)
		}
	}

	if len(filteredLengths) > 0 {
		t.printStatsForDataset(fmt.Sprintf("FILTERED DATA (≤ %d tokens)", maxTokens), filteredLengths, false)
	}
}

// printStatsForDataset prints statistics for a given dataset
func (t *Task6Tokenization) printStatsForDataset(title string, lengths []int32, showRetention bool) {
	// Calculate basic statistics
	count := int64(len(lengths))
	var sum int64
	for _, l := range lengths {
		sum += int64(l)
	}
	mean := float64(sum) / float64(count)

	// Calculate standard deviation
	var variance float64
	for _, l := range lengths {
		diff := float64(l) - mean
		variance += diff * diff
	}
	std := math.Sqrt(variance / float64(count))

	// Calculate percentiles
	percentile := func(p float64) int32 {
		idx := int(float64(count-1) * p)
		return lengths[idx]
	}

	min := lengths[0]
	p25 := percentile(0.25)
	p50 := percentile(0.50)
	p75 := percentile(0.75)
	p90 := percentile(0.90)
	p95 := percentile(0.95)
	p99 := percentile(0.99)
	max := lengths[len(lengths)-1]

	fmt.Printf("\n[TOKEN LENGTH STATISTICS - %s]\n", title)
	fmt.Printf("  Count:  %d records\n", count)
	fmt.Printf("  Sum:    %d tokens\n", sum)
	fmt.Printf("  Mean:   %.1f tokens\n", mean)
	fmt.Printf("  Std:    %.1f tokens\n", std)
	fmt.Printf("  Min:    %d tokens\n", min)
	fmt.Printf("  25%%:    %d tokens\n", p25)
	fmt.Printf("  50%%:    %d tokens (median)\n", p50)
	fmt.Printf("  75%%:    %d tokens\n", p75)
	fmt.Printf("  90%%:    %d tokens\n", p90)
	fmt.Printf("  95%%:    %d tokens\n", p95)
	fmt.Printf("  99%%:    %d tokens\n", p99)
	fmt.Printf("  Max:    %d tokens\n", max)

	// Retention analysis at common token length thresholds (only if requested)
	if showRetention {
		thresholds := []int32{8192, 16384, 32768, 65536, 131072}

		fmt.Println("\n[RETENTION ANALYSIS BY TOKEN LENGTH]")
		for _, threshold := range thresholds {
			var retainedCount int64
			var retainedTokens int64

			for _, l := range lengths {
				if l <= threshold {
					retainedCount++
					retainedTokens += int64(l)
				}
			}

			retentionPct := float64(retainedCount) / float64(count) * 100
			tokenPct := float64(retainedTokens) / float64(sum) * 100

			fmt.Printf("  <= %6d tokens: %9d records (%5.1f%%), %15d tokens (%5.1f%%)\n",
				threshold, retainedCount, retentionPct, retainedTokens, tokenPct)
		}
	}
}

// Close closes all writers and resources
func (t *Task6Tokenization) Close() error {
	var errs []error
	if err := t.writer.Close(); err != nil {
		errs = append(errs, fmt.Errorf("failed to close tokenized dataset writer: %w", err))
	}
	if err := t.statsWriter.Close(); err != nil {
		errs = append(errs, fmt.Errorf("failed to close statistics writer: %w", err))
	}

	// Note: Rust worker is closed in Run() after all responses are processed
	// This ensures the harvester can finish reading all responses

	if len(errs) > 0 {
		return fmt.Errorf("multiple errors closing task 6: %v", errs)
	}
	return nil
}
