init.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. package config
  2. import (
  3. "fmt"
  4. "os"
  5. "path/filepath"
  6. "slices"
  7. "strings"
  8. "sync/atomic"
  9. "github.com/charmbracelet/crush/internal/fsext"
  10. )
  11. const (
  12. InitFlagFilename = "init"
  13. )
  14. type ProjectInitFlag struct {
  15. Initialized bool `json:"initialized"`
  16. }
  17. // TODO: we need to remove the global config instance keeping it now just until everything is migrated
  18. var instance atomic.Pointer[Config]
  19. func Init(workingDir, dataDir string, debug bool) (*Config, error) {
  20. cfg, err := Load(workingDir, dataDir, debug)
  21. if err != nil {
  22. return nil, err
  23. }
  24. instance.Store(cfg)
  25. return instance.Load(), nil
  26. }
  27. func Get() *Config {
  28. cfg := instance.Load()
  29. return cfg
  30. }
  31. func ProjectNeedsInitialization() (bool, error) {
  32. cfg := Get()
  33. if cfg == nil {
  34. return false, fmt.Errorf("config not loaded")
  35. }
  36. flagFilePath := filepath.Join(cfg.Options.DataDirectory, InitFlagFilename)
  37. _, err := os.Stat(flagFilePath)
  38. if err == nil {
  39. return false, nil
  40. }
  41. if !os.IsNotExist(err) {
  42. return false, fmt.Errorf("failed to check init flag file: %w", err)
  43. }
  44. someContextFileExists, err := contextPathsExist(cfg.WorkingDir())
  45. if err != nil {
  46. return false, fmt.Errorf("failed to check for context files: %w", err)
  47. }
  48. if someContextFileExists {
  49. return false, nil
  50. }
  51. // If the working directory has no non-ignored files, skip initialization step
  52. empty, err := dirHasNoVisibleFiles(cfg.WorkingDir())
  53. if err != nil {
  54. return false, fmt.Errorf("failed to check if directory is empty: %w", err)
  55. }
  56. if empty {
  57. return false, nil
  58. }
  59. return true, nil
  60. }
  61. func contextPathsExist(dir string) (bool, error) {
  62. entries, err := os.ReadDir(dir)
  63. if err != nil {
  64. return false, err
  65. }
  66. // Create a slice of lowercase filenames for lookup with slices.Contains
  67. var files []string
  68. for _, entry := range entries {
  69. if !entry.IsDir() {
  70. files = append(files, strings.ToLower(entry.Name()))
  71. }
  72. }
  73. // Check if any of the default context paths exist in the directory
  74. for _, path := range defaultContextPaths {
  75. // Extract just the filename from the path
  76. _, filename := filepath.Split(path)
  77. filename = strings.ToLower(filename)
  78. if slices.Contains(files, filename) {
  79. return true, nil
  80. }
  81. }
  82. return false, nil
  83. }
  84. // dirHasNoVisibleFiles returns true if the directory has no files/dirs after applying ignore rules
  85. func dirHasNoVisibleFiles(dir string) (bool, error) {
  86. files, _, err := fsext.ListDirectory(dir, nil, 1, 1)
  87. if err != nil {
  88. return false, err
  89. }
  90. return len(files) == 0, nil
  91. }
  92. func MarkProjectInitialized() error {
  93. cfg := Get()
  94. if cfg == nil {
  95. return fmt.Errorf("config not loaded")
  96. }
  97. flagFilePath := filepath.Join(cfg.Options.DataDirectory, InitFlagFilename)
  98. file, err := os.Create(flagFilePath)
  99. if err != nil {
  100. return fmt.Errorf("failed to create init flag file: %w", err)
  101. }
  102. defer file.Close()
  103. return nil
  104. }
  105. func HasInitialDataConfig() bool {
  106. cfgPath := GlobalConfigData()
  107. if _, err := os.Stat(cfgPath); err != nil {
  108. return false
  109. }
  110. return Get().IsConfigured()
  111. }