Files
Fabric/internal/tools/patterns_loader.go
Kayvan Sylvan 7570e7930b feat: localize setup process and add funding configuration
- Add GitHub and Buy Me a Coffee funding configuration.
- Localize setup prompts and error messages across multiple languages.
- Implement helper for localized questions with static environment keys.
- Update environment variable builder to handle hyphenated plugin names.
- Replace hardcoded console output with localized i18n translation strings.
- Expand locale files with comprehensive pattern and strategy translations.
- Add new i18n keys for optional and required markers
- Remove hardcoded `[required]` markers from description strings
- Add custom patterns, Jina AI, YouTube, and language labels
- Switch plugin descriptions to use i18n translation keys
- Append markers dynamically to setup descriptions in Go code
- Remove trailing newlines from plugin question prompt strings
- Standardize all locale files with consistent formatting changes
2025-12-22 09:39:02 -08:00

372 lines
11 KiB
Go

package tools
import (
"fmt"
"os"
"path/filepath"
"sort"
"strings"
"github.com/danielmiessler/fabric/internal/i18n"
debuglog "github.com/danielmiessler/fabric/internal/log"
"github.com/danielmiessler/fabric/internal/plugins"
"github.com/danielmiessler/fabric/internal/plugins/db/fsdb"
"github.com/danielmiessler/fabric/internal/tools/githelper"
"github.com/otiai10/copy"
)
const DefaultPatternsGitRepoUrl = "https://github.com/danielmiessler/fabric.git"
const DefaultPatternsGitRepoFolder = "data/patterns"
func NewPatternsLoader(patterns *fsdb.PatternsEntity) (ret *PatternsLoader) {
label := "Patterns Loader"
ret = &PatternsLoader{
Patterns: patterns,
loadedFilePath: patterns.BuildFilePath("loaded"),
}
ret.PluginBase = &plugins.PluginBase{
Name: i18n.T("patterns_loader_label"),
SetupDescription: i18n.T("patterns_setup_description") + " " + i18n.T("required_marker"),
EnvNamePrefix: plugins.BuildEnvVariablePrefix(label),
ConfigureCustom: ret.configure,
}
ret.DefaultGitRepoUrl = ret.AddSetupQuestionWithEnvName("Git Repo Url", true,
i18n.T("patterns_git_repo_url_question"))
ret.DefaultGitRepoUrl.Value = DefaultPatternsGitRepoUrl
ret.DefaultFolder = ret.AddSetupQuestionWithEnvName("Git Repo Patterns Folder", true,
i18n.T("patterns_git_repo_folder_question"))
ret.DefaultFolder.Value = DefaultPatternsGitRepoFolder
return
}
type PatternsLoader struct {
*plugins.PluginBase
Patterns *fsdb.PatternsEntity
DefaultGitRepoUrl *plugins.SetupQuestion
DefaultFolder *plugins.SetupQuestion
loadedFilePath string
pathPatternsPrefix string
tempPatternsFolder string
}
func (o *PatternsLoader) configure() (err error) {
o.pathPatternsPrefix = fmt.Sprintf("%v/", o.DefaultFolder.Value)
// Use a consistent temp folder name regardless of the source path structure
tempDir, err := os.MkdirTemp("", "fabric-patterns-")
if err != nil {
return fmt.Errorf(i18n.T("patterns_failed_create_temp_folder"), err)
}
o.tempPatternsFolder = tempDir
return
}
func (o *PatternsLoader) IsConfigured() (ret bool) {
ret = o.PluginBase.IsConfigured()
if ret {
if _, err := os.Stat(o.loadedFilePath); os.IsNotExist(err) {
ret = false
}
}
return
}
func (o *PatternsLoader) Setup() (err error) {
if err = o.PluginBase.Setup(); err != nil {
return
}
if err = o.PopulateDB(); err != nil {
return
}
return
}
// PopulateDB downloads patterns from the internet and populates the patterns folder
func (o *PatternsLoader) PopulateDB() (err error) {
fmt.Printf(i18n.T("patterns_downloading"), o.Patterns.Dir)
fmt.Println()
fmt.Println()
originalPath := o.DefaultFolder.Value
if err = o.gitCloneAndCopy(); err != nil {
return fmt.Errorf(i18n.T("patterns_failed_download_from_git"), err)
}
// If the path was migrated during gitCloneAndCopy, we need to save the updated configuration
if o.DefaultFolder.Value != originalPath {
fmt.Printf(i18n.T("patterns_saving_updated_configuration"), originalPath, o.DefaultFolder.Value)
// The configuration will be saved by the calling code after this returns successfully
}
if err = o.movePatterns(); err != nil {
return fmt.Errorf(i18n.T("patterns_failed_move_patterns"), err)
}
fmt.Printf(i18n.T("patterns_download_success"), o.Patterns.Dir)
// Create the unique patterns file after patterns are successfully moved
if err = o.createUniquePatternsFile(); err != nil {
return fmt.Errorf(i18n.T("patterns_failed_unique_file"), err)
}
return
}
// PersistPatterns copies custom patterns to the updated patterns directory
func (o *PatternsLoader) PersistPatterns() (err error) {
// Check if patterns directory exists, if not, nothing to persist
if _, err = os.Stat(o.Patterns.Dir); err != nil {
if os.IsNotExist(err) {
// No existing patterns directory, nothing to persist
return nil
}
// Return unexpected errors (e.g., permission issues)
return fmt.Errorf(i18n.T("patterns_failed_access_directory"), o.Patterns.Dir, err)
}
var currentPatterns []os.DirEntry
if currentPatterns, err = os.ReadDir(o.Patterns.Dir); err != nil {
return
}
newPatternsFolder := o.tempPatternsFolder
var newPatterns []os.DirEntry
if newPatterns, err = os.ReadDir(newPatternsFolder); err != nil {
return
}
// Create a map of new patterns for faster lookup
newPatternNames := make(map[string]bool)
for _, newPattern := range newPatterns {
if newPattern.IsDir() {
newPatternNames[newPattern.Name()] = true
}
}
// Copy custom patterns that don't exist in the new download
for _, currentPattern := range currentPatterns {
if currentPattern.IsDir() && !newPatternNames[currentPattern.Name()] {
// This is a custom pattern, preserve it
src := filepath.Join(o.Patterns.Dir, currentPattern.Name())
dst := filepath.Join(newPatternsFolder, currentPattern.Name())
if copyErr := copy.Copy(src, dst); copyErr != nil {
fmt.Printf(i18n.T("patterns_preserve_warning"), currentPattern.Name(), copyErr)
} else {
fmt.Printf(i18n.T("patterns_preserved_custom_pattern"), currentPattern.Name())
}
}
}
return nil
}
// movePatterns copies the new patterns into the config directory
func (o *PatternsLoader) movePatterns() (err error) {
if err = os.MkdirAll(o.Patterns.Dir, os.ModePerm); err != nil {
return
}
patternsDir := o.tempPatternsFolder
if err = o.PersistPatterns(); err != nil {
return
}
if err = copy.Copy(patternsDir, o.Patterns.Dir); err != nil { // copies the patterns to the config directory
return
}
// Verify that patterns were actually copied before creating the loaded marker
var entries []os.DirEntry
if entries, err = os.ReadDir(o.Patterns.Dir); err != nil {
return
}
// Count actual pattern directories (exclude the loaded file itself)
patternCount := 0
for _, entry := range entries {
if entry.IsDir() {
patternCount++
}
}
if patternCount == 0 {
err = fmt.Errorf(i18n.T("patterns_no_patterns_copied"), o.Patterns.Dir)
return
}
//create an empty file to indicate that the patterns have been updated if not exists
if _, err = os.Create(o.loadedFilePath); err != nil {
return fmt.Errorf(i18n.T("patterns_failed_loaded_marker"), o.loadedFilePath, err)
}
err = os.RemoveAll(patternsDir)
return
}
func (o *PatternsLoader) gitCloneAndCopy() (err error) {
// Create temp folder if it doesn't exist
if err = os.MkdirAll(filepath.Dir(o.tempPatternsFolder), os.ModePerm); err != nil {
return fmt.Errorf(i18n.T("patterns_failed_create_temp_dir"), err)
}
fmt.Printf(i18n.T("patterns_cloning_repository"), o.DefaultGitRepoUrl.Value, o.DefaultFolder.Value)
// Try to fetch files with the current path
err = githelper.FetchFilesFromRepo(githelper.FetchOptions{
RepoURL: o.DefaultGitRepoUrl.Value,
PathPrefix: o.DefaultFolder.Value,
DestDir: o.tempPatternsFolder,
})
if err != nil {
return fmt.Errorf(i18n.T("patterns_failed_download_from_repo"), o.DefaultGitRepoUrl.Value, err)
}
// Check if patterns were downloaded
if patternCount, checkErr := o.countPatternsInDirectory(o.tempPatternsFolder); checkErr != nil {
return fmt.Errorf(i18n.T("patterns_failed_read_temp_directory"), checkErr)
} else if patternCount == 0 {
// No patterns found with current path, try automatic migration
if migrationErr := o.tryPathMigration(); migrationErr != nil {
return fmt.Errorf(i18n.T("patterns_no_patterns_migration_failed"), o.DefaultFolder.Value, migrationErr)
}
// Migration successful, try downloading again
return o.gitCloneAndCopy()
} else {
fmt.Printf(i18n.T("patterns_downloaded_temp"), patternCount)
}
return nil
}
// tryPathMigration attempts to migrate from old pattern paths to new restructured paths
func (o *PatternsLoader) tryPathMigration() (err error) {
// Check if current path is the old "patterns" path
if o.DefaultFolder.Value == "patterns" {
fmt.Println(i18n.T("patterns_detected_old_path"))
// Try the new restructured path
newPath := "data/patterns"
testTempFolder := filepath.Join(os.TempDir(), "fabric-patterns-test")
// Clean up any existing test temp folder
if err := os.RemoveAll(testTempFolder); err != nil {
fmt.Printf(i18n.T("patterns_warning_remove_test_folder"), testTempFolder, err)
}
// Test if the new path works
testErr := githelper.FetchFilesFromRepo(githelper.FetchOptions{
RepoURL: o.DefaultGitRepoUrl.Value,
PathPrefix: newPath,
DestDir: testTempFolder,
})
if testErr == nil {
// Check if patterns exist in the new path
if patternCount, countErr := o.countPatternsInDirectory(testTempFolder); countErr == nil && patternCount > 0 {
fmt.Printf(i18n.T("patterns_found_new_path"), patternCount, newPath)
// Update the configuration
o.DefaultFolder.Value = newPath
// Clean up the main temp folder and replace it with the test one
os.RemoveAll(o.tempPatternsFolder)
if renameErr := os.Rename(testTempFolder, o.tempPatternsFolder); renameErr != nil {
// If rename fails, try copy
if copyErr := copy.Copy(testTempFolder, o.tempPatternsFolder); copyErr != nil {
return fmt.Errorf(i18n.T("patterns_failed_move_test_patterns"), copyErr)
}
os.RemoveAll(testTempFolder)
}
return nil
}
}
// Clean up test folder
os.RemoveAll(testTempFolder)
}
return fmt.Errorf(i18n.T("patterns_unable_to_find_or_migrate"), o.DefaultFolder.Value)
}
// countPatternsInDirectory counts the number of pattern directories in a given directory
func (o *PatternsLoader) countPatternsInDirectory(dir string) (int, error) {
entries, err := os.ReadDir(dir)
if err != nil {
return 0, err
}
patternCount := 0
for _, entry := range entries {
if entry.IsDir() {
patternCount++
}
}
return patternCount, nil
}
// createUniquePatternsFile creates the unique_patterns.txt file with all pattern names
func (o *PatternsLoader) createUniquePatternsFile() (err error) {
// Read patterns from the main patterns directory
entries, err := os.ReadDir(o.Patterns.Dir)
if err != nil {
return fmt.Errorf(i18n.T("patterns_failed_read_directory"), err)
}
patternNamesMap := make(map[string]bool) // Use map to avoid duplicates
// Add patterns from main directory
for _, entry := range entries {
if entry.IsDir() {
patternNamesMap[entry.Name()] = true
}
}
// Add patterns from custom patterns directory if it exists
if o.Patterns.CustomPatternsDir != "" {
if customEntries, customErr := os.ReadDir(o.Patterns.CustomPatternsDir); customErr == nil {
for _, entry := range customEntries {
if entry.IsDir() {
patternNamesMap[entry.Name()] = true
}
}
debuglog.Log(i18n.T("patterns_debug_included_custom_directory"), o.Patterns.CustomPatternsDir)
} else {
debuglog.Log(i18n.T("patterns_warning_custom_directory"), o.Patterns.CustomPatternsDir, customErr)
}
}
if len(patternNamesMap) == 0 {
if o.Patterns.CustomPatternsDir != "" {
return fmt.Errorf(i18n.T("patterns_no_patterns_found_in_directories"), o.Patterns.Dir, o.Patterns.CustomPatternsDir)
}
return fmt.Errorf(i18n.T("patterns_no_patterns_found_in_directory"), o.Patterns.Dir)
}
// Convert map to sorted slice
var patternNames []string
for name := range patternNamesMap {
patternNames = append(patternNames, name)
}
// Sort patterns alphabetically for consistent output
sort.Strings(patternNames)
// Join pattern names with newlines
content := strings.Join(patternNames, "\n") + "\n"
if err = os.WriteFile(o.Patterns.UniquePatternsFilePath, []byte(content), 0644); err != nil {
return fmt.Errorf(i18n.T("patterns_failed_write_unique_file"), err)
}
fmt.Printf(i18n.T("patterns_unique_file_created"), len(patternNames))
return nil
}