9240a863d1
Part 1: Shared Utilities - Create internal/fileutil package with FindDataFile() and LoadJSON() - Create internal/lang package with language constants and validation - Eliminate 46 lines of code duplication between cv/loader.go and ui/loader.go - Simplify cv/loader.go from 69 to 36 lines (-48%) - Simplify ui/loader.go from 56 to 24 lines (-57%) Part 2: Validation Layer - Add comprehensive validation in internal/models/cv/validation.go - Validate Personal (name, email format, URLs) - Validate Experience (required fields, dates) - Validate Education (required fields) - Validate Skills (proficiency ranges 1-5, categories) - Validate Languages (proficiency levels 1-5) - Validate Projects (title, URLs) - Validate Meta (version, language) - Integrate validation into LoadCV() - automatic on load - Create ValidationError and ValidationErrors types for clear error reporting - Report all validation errors at once (better UX) Testing: - Add comprehensive tests for fileutil package (FindDataFile, LoadJSON) - Add tests for lang package (IsValid, Validate, All) - Add 280+ validation test cases covering edge cases - All tests pass with real CV data (cv-en.json, cv-es.json) - Fixed validation to allow both URLs and local paths for gitRepoUrl Documentation: - Create _go-learning/refactorings/002-shared-utilities-validation.md - Document architecture, benefits, testing, and interview talking points - Explain WHY decisions were made (DRY, type safety, data integrity) Benefits: - DRY: Single source of truth for utilities - Type safety: Language constants instead of magic strings - Data integrity: Validation catches errors at load time - Better errors: Clear messages showing all issues at once - Maintainability: Centralized utilities easier to update
42 lines
1.0 KiB
Go
42 lines
1.0 KiB
Go
package cv
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/juanatsap/cv-site/internal/fileutil"
|
|
"github.com/juanatsap/cv-site/internal/lang"
|
|
)
|
|
|
|
// LoadCV loads CV data from a JSON file for the specified language
|
|
func LoadCV(language string) (*CV, error) {
|
|
if err := lang.Validate(language); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var cvData CV
|
|
filename := fmt.Sprintf("data/cv-%s.json", language)
|
|
if err := fileutil.LoadJSON(filename, &cvData); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Replace {{YEAR}} placeholder in reference URLs with current year
|
|
currentYear := fmt.Sprintf("%d", time.Now().Year())
|
|
for i := range cvData.References {
|
|
cvData.References[i].URL = replaceYearPlaceholder(cvData.References[i].URL, currentYear)
|
|
}
|
|
|
|
// Validate the loaded CV data
|
|
if err := cvData.Validate(); err != nil {
|
|
return nil, fmt.Errorf("validation failed: %w", err)
|
|
}
|
|
|
|
return &cvData, nil
|
|
}
|
|
|
|
// replaceYearPlaceholder replaces {{YEAR}} with the current year
|
|
func replaceYearPlaceholder(url string, year string) string {
|
|
return strings.ReplaceAll(url, "{{YEAR}}", year)
|
|
}
|