Compare commits
	
		
			27 Commits
		
	
	
		
			33b3a3d2b6
			...
			v1.5.2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| dbd736ae81 | |||
| ff76a5399c | |||
| 3f0791466b | |||
| dc5eb9cb80 | |||
| 25a8e2b65a | |||
| 59faaa181d | |||
| 7bff91679d | |||
| cfa7fc73c9 | |||
| a568a736aa | |||
| db72688aa2 | |||
| 05082d8ff3 | |||
| a4f90c2bc8 | |||
| bec5b3cb9c | |||
| 018c0797f5 | |||
| a7d5317114 | |||
| 89e29eacee | |||
| 4e4e58af83 | |||
| eb81ec4162 | |||
| 21d7f56ccf | |||
| 8653191df2 | |||
| 9da47ce0cf | |||
| 29bfa2d776 | |||
| c94a7ae8ab | |||
| ca57ee728e | |||
| b53628e698 | |||
| 3f7fd36f84 | |||
| 8da1f023a7 | 
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -3,3 +3,6 @@ | |||||||
| cln | cln | ||||||
| cln.log | cln.log | ||||||
| .qodo | .qodo | ||||||
|  | *.log | ||||||
|  | *.out | ||||||
|  | test_temp | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | { | ||||||
|  | 	// Use IntelliSense to learn about possible attributes. | ||||||
|  | 	// Hover to view descriptions of existing attributes. | ||||||
|  | 	// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 | ||||||
|  | 	"version": "0.2.0", | ||||||
|  | 	"configurations": [ | ||||||
|  | 		{ | ||||||
|  | 			"name": "Ereshor Workspace", | ||||||
|  | 			"type": "go", | ||||||
|  | 			"request": "launch", | ||||||
|  | 			"mode": "auto", | ||||||
|  | 			"program": "${workspaceFolder}", | ||||||
|  | 			"cwd": "C:\\Users\\Administrator\\Seafile\\Games-Ereshor" | ||||||
|  | 		} | ||||||
|  | 	] | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								build.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								build.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | |||||||
|  | CGO_ENABLED=0 GOOS=windows GOARCH=amd64 go build -o cln.exe . | ||||||
|  | CGO_ENABLED=0 GOOS=linux GOARCH=amd64 go build -o cln . | ||||||
| @@ -83,6 +83,7 @@ const ( | |||||||
| // The acceptable range is [16, 231] but here we remove some very dark colors | // The acceptable range is [16, 231] but here we remove some very dark colors | ||||||
| // That make text unreadable on a dark terminal | // That make text unreadable on a dark terminal | ||||||
| // See https://www.hackitu.de/termcolor256/ | // See https://www.hackitu.de/termcolor256/ | ||||||
|  | // Wait - why are we hardcoding this? lol do for loops not exist in our universe? | ||||||
| var colors = []int{22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 57, 62, 63, 64, 65, 67, 68, 69, 70, 71, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98, 99, 100, 101, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 124, 125, 126, 127, 128, 129, 130, 131, 132, 133, 134, 135, 136, 137, 138, 139, 140, 141, 142, 143, 144, 148, 149, 150, 151, 152, 154, 155, 156, 157, 158, 159, 160, 161, 162, 163, 164, 165, 166, 167, 168, 169, 170, 171, 172, 173, 174, 175, 176, 177, 178, 179, 180, 181, 182, 184, 185, 186, 190, 191, 192, 193, 194, 195, 196, 197, 198, 199, 200, 201, 202, 203, 204, 205, 206, 207, 208, 209, 210, 211, 212, 213, 214, 215, 216, 217, 218, 219, 220, 221, 222, 223, 226, 227, 228, 229, 230} | var colors = []int{22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 57, 62, 63, 64, 65, 67, 68, 69, 70, 71, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93, 94, 95, 96, 97, 98, 99, 100, 101, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 124, 125, 126, 127, 128, 129, 130, 131, 132, 133, 134, 135, 136, 137, 138, 139, 140, 141, 142, 143, 144, 148, 149, 150, 151, 152, 154, 155, 156, 157, 158, 159, 160, 161, 162, 163, 164, 165, 166, 167, 168, 169, 170, 171, 172, 173, 174, 175, 176, 177, 178, 179, 180, 181, 182, 184, 185, 186, 190, 191, 192, 193, 194, 195, 196, 197, 198, 199, 200, 201, 202, 203, 204, 205, 206, 207, 208, 209, 210, 211, 212, 213, 214, 215, 216, 217, 218, 219, 220, 221, 222, 223, 226, 227, 228, 229, 230} | ||||||
| var colorsIndex int = -1 | var colorsIndex int = -1 | ||||||
| var shuffled bool | var shuffled bool | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,7 +1,17 @@ | |||||||
| module cln | module cln | ||||||
|  |  | ||||||
| go 1.21.7 | go 1.23.6 | ||||||
|  |  | ||||||
| require gopkg.in/yaml.v3 v3.0.1 | require gopkg.in/yaml.v3 v3.0.1 | ||||||
|  |  | ||||||
| require github.com/bmatcuk/doublestar/v4 v4.8.1 | require ( | ||||||
|  | 	git.site.quack-lab.dev/dave/cyutils v1.4.0 | ||||||
|  | 	github.com/bmatcuk/doublestar/v4 v4.8.1 | ||||||
|  | 	github.com/stretchr/testify v1.11.1 | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | require ( | ||||||
|  | 	github.com/davecgh/go-spew v1.1.1 // indirect | ||||||
|  | 	github.com/pmezard/go-difflib v1.0.0 // indirect | ||||||
|  | 	golang.org/x/time v0.12.0 // indirect | ||||||
|  | ) | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,5 +1,15 @@ | |||||||
|  | git.site.quack-lab.dev/dave/cyutils v1.4.0 h1:/Xo3QfLIFNab+axHneWmUK4MyfuObl+qq8whF9vTQpk= | ||||||
|  | git.site.quack-lab.dev/dave/cyutils v1.4.0/go.mod h1:fBjALu2Cp2u2bDr+E4zbGVMBeIgFzROg+4TCcTNAiQU= | ||||||
| github.com/bmatcuk/doublestar/v4 v4.8.1 h1:54Bopc5c2cAvhLRAzqOGCYHYyhcDHsFF4wWIR5wKP38= | github.com/bmatcuk/doublestar/v4 v4.8.1 h1:54Bopc5c2cAvhLRAzqOGCYHYyhcDHsFF4wWIR5wKP38= | ||||||
| github.com/bmatcuk/doublestar/v4 v4.8.1/go.mod h1:xBQ8jztBU6kakFMg+8WGxn0c6z1fTSPVIjEY1Wr7jzc= | github.com/bmatcuk/doublestar/v4 v4.8.1/go.mod h1:xBQ8jztBU6kakFMg+8WGxn0c6z1fTSPVIjEY1Wr7jzc= | ||||||
|  | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||||
|  | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
|  | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
|  | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
|  | github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U= | ||||||
|  | github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U= | ||||||
|  | golang.org/x/time v0.12.0 h1:ScB/8o8olJvc+CQPWrK3fPZNfh7qgwCrY0zJmoEQLSE= | ||||||
|  | golang.org/x/time v0.12.0/go.mod h1:CDIdPxbZBQxdj6cxyCIdrNogrJKMJ7pr37NYpMcMDSg= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= | ||||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= | ||||||
|   | |||||||
							
								
								
									
										49
									
								
								home_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								home_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | |||||||
|  | package main | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestHomeDirectoryPatternExpansion(t *testing.T) { | ||||||
|  | 	testDir := getTestSubDir(t) | ||||||
|  |  | ||||||
|  | 	// Ensure we're working within the project directory | ||||||
|  | 	ensureInProjectDir(t, testDir) | ||||||
|  |  | ||||||
|  | 	// Change to test directory | ||||||
|  | 	originalDir, _ := os.Getwd() | ||||||
|  | 	defer os.Chdir(originalDir) | ||||||
|  | 	os.Chdir(testDir) | ||||||
|  |  | ||||||
|  | 	// Get the actual home directory | ||||||
|  | 	homeDir, err := os.UserHomeDir() | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 	// Create a test directory in the home folder | ||||||
|  | 	testHomeDir := filepath.Join(homeDir, "synclib_test") | ||||||
|  | 	err = os.MkdirAll(testHomeDir, 0755) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	defer os.RemoveAll(testHomeDir) // Cleanup | ||||||
|  |  | ||||||
|  | 	// Create a test file in the home directory | ||||||
|  | 	testFile := filepath.Join(testHomeDir, "testhome.csv") | ||||||
|  | 	err = os.WriteFile(testFile, []byte("test content"), 0644) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  |  | ||||||
|  | 	// Test the pattern with ~/ that should match the file | ||||||
|  | 	pattern := "~/synclib_test/testhome.csv" | ||||||
|  | 	links, err := ExpandPattern(pattern, testDir, "target.csv") | ||||||
|  |  | ||||||
|  | 	// This should work but currently fails due to the bug | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, 1, len(links), "Pattern should match exactly 1 file") | ||||||
|  |  | ||||||
|  | 	if len(links) > 0 { | ||||||
|  | 		assert.Contains(t, links[0].Source, "testhome.csv") | ||||||
|  | 		assert.Equal(t, "target.csv", links[0].Target) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										214
									
								
								instruction.go
									
									
									
									
									
								
							
							
						
						
									
										214
									
								
								instruction.go
									
									
									
									
									
								
							| @@ -20,6 +20,7 @@ type LinkInstruction struct { | |||||||
|  |  | ||||||
| type YAMLConfig struct { | type YAMLConfig struct { | ||||||
| 	Links []LinkInstruction `yaml:"links"` | 	Links []LinkInstruction `yaml:"links"` | ||||||
|  | 	From  []string          `yaml:"from,omitempty"` | ||||||
| } | } | ||||||
|  |  | ||||||
| func (instruction *LinkInstruction) Tidy() { | func (instruction *LinkInstruction) Tidy() { | ||||||
| @@ -269,40 +270,24 @@ func ParseYAMLFile(filename, workdir string) ([]LinkInstruction, error) { | |||||||
| 		return nil, fmt.Errorf("error reading YAML file: %w", err) | 		return nil, fmt.Errorf("error reading YAML file: %w", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// First try to parse as a list of link instructions | 	// Parse as a direct list of instructions | ||||||
| 	var config YAMLConfig | 	var instructions []LinkInstruction | ||||||
| 	err = yaml.Unmarshal(data, &config) | 	err = yaml.Unmarshal(data, &instructions) | ||||||
| 	if err != nil || len(config.Links) == 0 { | 	if err != nil { | ||||||
| 		// If that fails, try parsing as a direct list of instructions | 		return nil, fmt.Errorf("error parsing YAML: %w", err) | ||||||
| 		var instructions []LinkInstruction |  | ||||||
| 		err = yaml.Unmarshal(data, &instructions) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return nil, fmt.Errorf("error parsing YAML: %w", err) |  | ||||||
| 		} |  | ||||||
| 		config.Links = instructions |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	expanded := []LinkInstruction{} | 	// Preprocess instructions: expand globs and from references | ||||||
| 	for _, link := range config.Links { | 	// Create a new visited map for this file | ||||||
| 		LogSource("Expanding pattern source %s in YAML file %s", link.Source, filename) | 	visited := make(map[string]bool) | ||||||
| 		newlinks, err := ExpandPattern(link.Source, workdir, link.Target) | 	processedInstructions, err := preprocessInstructions(instructions, filename, workdir, visited) | ||||||
| 		if err != nil { | 	if err != nil { | ||||||
| 			return nil, fmt.Errorf("error expanding pattern: %w", err) | 		return nil, err | ||||||
| 		} |  | ||||||
| 		// "Clone" the original link instruction for each expanded link |  | ||||||
| 		for i := range newlinks { |  | ||||||
| 			newlinks[i].Delete = link.Delete |  | ||||||
| 			newlinks[i].Hard = link.Hard |  | ||||||
| 			newlinks[i].Force = link.Force |  | ||||||
| 		} |  | ||||||
| 		LogInfo("Expanded pattern %s in YAML file %s to %d links", |  | ||||||
| 			FormatSourcePath(link.Source), FormatSourcePath(filename), len(newlinks)) |  | ||||||
| 		expanded = append(expanded, newlinks...) |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for i := range expanded { | 	// Final processing: normalize paths and set defaults | ||||||
| 		link := &expanded[i] | 	for i := range processedInstructions { | ||||||
|  | 		link := &processedInstructions[i] | ||||||
| 		link.Tidy() | 		link.Tidy() | ||||||
| 		link.Source, _ = ConvertHome(link.Source) | 		link.Source, _ = ConvertHome(link.Source) | ||||||
| 		link.Target, _ = ConvertHome(link.Target) | 		link.Target, _ = ConvertHome(link.Target) | ||||||
| @@ -315,12 +300,153 @@ func ParseYAMLFile(filename, workdir string) ([]LinkInstruction, error) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return expanded, nil | 	return processedInstructions, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // preprocessInstructions handles glob expansion and from references | ||||||
|  | func preprocessInstructions(instructions []LinkInstruction, filename, workdir string, visited map[string]bool) ([]LinkInstruction, error) { | ||||||
|  | 	var result []LinkInstruction | ||||||
|  |  | ||||||
|  | 	for _, instr := range instructions { | ||||||
|  | 		if instr.Source == "" { | ||||||
|  | 			continue // Skip invalid instructions | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if instr.Target == "" { | ||||||
|  | 			// This is a from reference - load the referenced file | ||||||
|  | 			fromInstructions, err := loadFromReference(instr.Source, filename, workdir, visited) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, fmt.Errorf("error loading from reference %s: %w", instr.Source, err) | ||||||
|  | 			} | ||||||
|  | 			result = append(result, fromInstructions...) | ||||||
|  | 		} else { | ||||||
|  | 			// This is a regular instruction - expand globs if needed | ||||||
|  | 			expandedInstructions, err := expandGlobs(instr, filename, workdir) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, fmt.Errorf("error expanding globs for %s: %w", instr.Source, err) | ||||||
|  | 			} | ||||||
|  | 			result = append(result, expandedInstructions...) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return result, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // loadFromReference loads instructions from a referenced file | ||||||
|  | func loadFromReference(fromFile, currentFile, workdir string, visited map[string]bool) ([]LinkInstruction, error) { | ||||||
|  | 	// First convert home directory if it starts with ~ | ||||||
|  | 	fromPath, err := ConvertHome(fromFile) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("error converting home directory: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Convert relative paths to absolute paths based on the current file's directory | ||||||
|  | 	if !filepath.IsAbs(fromPath) { | ||||||
|  | 		currentDir := filepath.Dir(currentFile) | ||||||
|  | 		fromPath = filepath.Join(currentDir, fromPath) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Normalize the path | ||||||
|  | 	fromPath = filepath.Clean(fromPath) | ||||||
|  |  | ||||||
|  | 	// Recursively parse the referenced file with cycle detection | ||||||
|  | 	fromWorkdir := filepath.Dir(fromPath) | ||||||
|  | 	return parseYAMLFileRecursive(fromPath, fromWorkdir, visited) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // expandGlobs expands glob patterns in a single instruction | ||||||
|  | func expandGlobs(instr LinkInstruction, filename, workdir string) ([]LinkInstruction, error) { | ||||||
|  | // Convert home directory (~) before expanding pattern | ||||||
|  | 	convertedSource, err := ConvertHome(instr.Source) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("error converting home directory in source %s: %w", instr.Source, err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	LogSource("Expanding pattern source %s in YAML file %s", convertedSource, filename) | ||||||
|  | 	newlinks, err := ExpandPattern(convertedSource, workdir, instr.Target) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Clone the original instruction properties for each expanded link | ||||||
|  | 	for i := range newlinks { | ||||||
|  | 		newlinks[i].Delete = instr.Delete | ||||||
|  | 		newlinks[i].Hard = instr.Hard | ||||||
|  | 		newlinks[i].Force = instr.Force | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	LogInfo("Expanded pattern %s in YAML file %s to %d links", | ||||||
|  | 		FormatSourcePath(instr.Source), FormatSourcePath(filename), len(newlinks)) | ||||||
|  |  | ||||||
|  | 	return newlinks, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ParseYAMLFileRecursive parses a YAML file and recursively processes any "From" references | ||||||
|  | func ParseYAMLFileRecursive(filename, workdir string) ([]LinkInstruction, error) { | ||||||
|  | 	visited := make(map[string]bool) | ||||||
|  | 	return parseYAMLFileRecursive(filename, workdir, visited) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // parseYAMLFileRecursive is the internal recursive function that tracks visited files to prevent cycles | ||||||
|  | func parseYAMLFileRecursive(filename, workdir string, visited map[string]bool) ([]LinkInstruction, error) { | ||||||
|  | 	// Normalize the filename to prevent cycles with different path representations | ||||||
|  | 	normalizedFilename, err := filepath.Abs(filename) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("error normalizing filename: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Check for cycles | ||||||
|  | 	if visited[normalizedFilename] { | ||||||
|  | 		return nil, fmt.Errorf("circular reference detected: %s", filename) | ||||||
|  | 	} | ||||||
|  | 	visited[normalizedFilename] = true | ||||||
|  | 	defer delete(visited, normalizedFilename) | ||||||
|  |  | ||||||
|  | 	// Parse the current file and preprocess it with cycle detection | ||||||
|  | 	data, err := os.ReadFile(filename) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("error reading YAML file: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Parse as a direct list of instructions | ||||||
|  | 	var instructions []LinkInstruction | ||||||
|  | 	err = yaml.Unmarshal(data, &instructions) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("error parsing YAML: %w", err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Preprocess instructions: expand globs and from references | ||||||
|  | 	processedInstructions, err := preprocessInstructions(instructions, filename, workdir, visited) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Final processing: normalize paths and set defaults | ||||||
|  | 	for i := range processedInstructions { | ||||||
|  | 		link := &processedInstructions[i] | ||||||
|  | 		link.Tidy() | ||||||
|  | 		link.Source, _ = ConvertHome(link.Source) | ||||||
|  | 		link.Target, _ = ConvertHome(link.Target) | ||||||
|  | 		link.Source = NormalizePath(link.Source, workdir) | ||||||
|  | 		link.Target = NormalizePath(link.Target, workdir) | ||||||
|  |  | ||||||
|  | 		// If Delete is true, Force must also be true | ||||||
|  | 		if link.Delete { | ||||||
|  | 			link.Force = true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return processedInstructions, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func ExpandPattern(source, workdir, target string) (links []LinkInstruction, err error) { | func ExpandPattern(source, workdir, target string) (links []LinkInstruction, err error) { | ||||||
|  | 	// Convert home directory (~) before splitting pattern | ||||||
|  | 	source, err = ConvertHome(source) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("error converting home directory in source %s: %w", source, err) | ||||||
|  | 	} | ||||||
| 	static, pattern := doublestar.SplitPattern(source) | 	static, pattern := doublestar.SplitPattern(source) | ||||||
| 	if static == "" { | 	if static == "" || static == "." { | ||||||
| 		static = workdir | 		static = workdir | ||||||
| 	} | 	} | ||||||
| 	LogInfo("Static part: %s", static) | 	LogInfo("Static part: %s", static) | ||||||
| @@ -337,21 +463,21 @@ func ExpandPattern(source, workdir, target string) (links []LinkInstruction, err | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, file := range files { | 	for _, file := range files { | ||||||
|  | 		if len(files) == 1 { | ||||||
|  | 			// Special case: if there is only one file | ||||||
|  | 			// This should only ever happen if our source is a path (and not a glob!) | ||||||
|  | 			// And our target is a path | ||||||
|  | 			// ...but it will also happen if the source IS a glob and it happens to match ONE file | ||||||
|  | 			// I think that should happen rarely enough to not be an issue... | ||||||
|  | 			links = append(links, LinkInstruction{ | ||||||
|  | 				Source: filepath.Join(static, file), | ||||||
|  | 				Target: target, | ||||||
|  | 			}) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
| 		if info, err := os.Stat(file); err == nil && info.IsDir() { | 		if info, err := os.Stat(file); err == nil && info.IsDir() { | ||||||
| 			// We don't care about matched directories | 			// We don't care about matched directories | ||||||
| 			// We want files within them | 			// We want files within them | ||||||
| 			if len(files) == 1 { |  | ||||||
| 				// Special case: if there is only one file, and it's a directory |  | ||||||
| 				// This should only ever happen if our source is a path (and not a glob!) |  | ||||||
| 				// And our target is a path, a directory |  | ||||||
| 				// ...but it will also happen if the source IS a glob and it happens to match ONE directory |  | ||||||
| 				// I think that should happen rarely enough to not be an issue... |  | ||||||
| 				links = append(links, LinkInstruction{ |  | ||||||
| 					Source: filepath.Join(static, file), |  | ||||||
| 					Target: target, |  | ||||||
| 				}) |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			LogInfo("Skipping directory %s", file) | 			LogInfo("Skipping directory %s", file) | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
|   | |||||||
							
								
								
									
										4441
									
								
								instruction_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4441
									
								
								instruction_test.go
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										224
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										224
									
								
								main.go
									
									
									
									
									
								
							| @@ -7,9 +7,8 @@ import ( | |||||||
| 	"log" | 	"log" | ||||||
| 	"os" | 	"os" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"regexp" |  | ||||||
| 	"sync" |  | ||||||
| 	"sync/atomic" | 	"sync/atomic" | ||||||
|  | 	utils "git.site.quack-lab.dev/dave/cyutils" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| const deliminer = "," | const deliminer = "," | ||||||
| @@ -20,7 +19,6 @@ const ImportantColor = BRed | |||||||
| const DefaultColor = Reset | const DefaultColor = Reset | ||||||
| const PathColor = Green | const PathColor = Green | ||||||
|  |  | ||||||
| var FileRegex, _ = regexp.Compile(`sync\.ya?ml$`) |  | ||||||
| var programName = os.Args[0] | var programName = os.Args[0] | ||||||
| var undo = false | var undo = false | ||||||
|  |  | ||||||
| @@ -32,7 +30,27 @@ func main() { | |||||||
| 	flag.Parse() | 	flag.Parse() | ||||||
| 	undo = *undoF | 	undo = *undoF | ||||||
|  |  | ||||||
| 	if *debug { | 	setupLogging(*debug) | ||||||
|  |  | ||||||
|  | 	instructions := make(chan *LinkInstruction, 1000) | ||||||
|  | 	status := make(chan error) | ||||||
|  |  | ||||||
|  | 	startInputSource(*recurse, *file, instructions, status) | ||||||
|  |  | ||||||
|  | 	go handleStatusErrors(status) | ||||||
|  |  | ||||||
|  | 	instructionsDone := processInstructions(instructions) | ||||||
|  |  | ||||||
|  | 	if instructionsDone == 0 { | ||||||
|  | 		LogInfo("No instructions were processed") | ||||||
|  | 		os.Exit(1) | ||||||
|  | 	} | ||||||
|  | 	LogInfo("All done") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setupLogging configures logging based on debug flag | ||||||
|  | func setupLogging(debug bool) { | ||||||
|  | 	if debug { | ||||||
| 		log.SetFlags(log.Lmicroseconds | log.Lshortfile) | 		log.SetFlags(log.Lmicroseconds | log.Lshortfile) | ||||||
| 		logFile, err := os.Create(programName + ".log") | 		logFile, err := os.Create(programName + ".log") | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| @@ -44,19 +62,19 @@ func main() { | |||||||
| 	} else { | 	} else { | ||||||
| 		log.SetFlags(log.Lmicroseconds) | 		log.SetFlags(log.Lmicroseconds) | ||||||
| 	} | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| 	instructions := make(chan *LinkInstruction, 1000) | // startInputSource determines and starts the appropriate input source | ||||||
| 	status := make(chan error) | func startInputSource(recurse, file string, instructions chan *LinkInstruction, status chan error) { | ||||||
|  |  | ||||||
| 	// Check input sources in priority order | 	// Check input sources in priority order | ||||||
| 	switch { | 	switch { | ||||||
| 	case *recurse != "": | 	case recurse != "": | ||||||
| 		LogInfo("Recurse: %s", *recurse) | 		LogInfo("Recurse: %s", recurse) | ||||||
| 		go ReadFromFilesRecursively(*recurse, instructions, status) | 		go ReadFromFilesRecursively(recurse, instructions, status) | ||||||
|  |  | ||||||
| 	case *file != "": | 	case file != "": | ||||||
| 		LogInfo("File: %s", *file) | 		LogInfo("File: %s", file) | ||||||
| 		go ReadFromFile(*file, instructions, status, true) | 		go ReadFromFile(file, instructions, status, true) | ||||||
|  |  | ||||||
| 	case len(flag.Args()) > 0: | 	case len(flag.Args()) > 0: | ||||||
| 		LogInfo("Reading from command line arguments") | 		LogInfo("Reading from command line arguments") | ||||||
| @@ -67,64 +85,81 @@ func main() { | |||||||
| 	// 	go ReadFromStdin(instructions, status) | 	// 	go ReadFromStdin(instructions, status) | ||||||
|  |  | ||||||
| 	default: | 	default: | ||||||
| 		if _, err := os.Stat("sync"); err == nil { | 		startDefaultInputSource(instructions, status) | ||||||
| 			LogInfo("Using default sync file") | 	} | ||||||
| 			go ReadFromFile("sync", instructions, status, true) | } | ||||||
| 		} else if _, err := os.Stat("sync.yaml"); err == nil { |  | ||||||
| 			LogInfo("Using default sync.yaml file") | // startDefaultInputSource tries to find default sync files | ||||||
| 			go ReadFromFile("sync.yaml", instructions, status, true) | func startDefaultInputSource(instructions chan *LinkInstruction, status chan error) { | ||||||
| 		} else if _, err := os.Stat("sync.yml"); err == nil { | 	if _, err := os.Stat("sync"); err == nil { | ||||||
| 			LogInfo("Using default sync.yml file") | 		LogInfo("Using default sync file") | ||||||
| 			go ReadFromFile("sync.yml", instructions, status, true) | 		go ReadFromFile("sync", instructions, status, true) | ||||||
| 		} else { | 	} else if _, err := os.Stat("sync.yaml"); err == nil { | ||||||
| 			LogInfo("No input provided") | 		LogInfo("Using default sync.yaml file") | ||||||
| 			LogInfo("Provide input as: ") | 		go ReadFromFile("sync.yaml", instructions, status, true) | ||||||
| 			LogInfo("Arguments - %s <source>,<target>,<force?>", programName) | 	} else if _, err := os.Stat("sync.yml"); err == nil { | ||||||
| 			LogInfo("File - %s -f <file>", programName) | 		LogInfo("Using default sync.yml file") | ||||||
| 			LogInfo("YAML File - %s -f <file.yaml>", programName) | 		go ReadFromFile("sync.yml", instructions, status, true) | ||||||
| 			LogInfo("Folder (finding sync files in folder recursively) - %s -r <folder>", programName) | 	} else { | ||||||
| 			LogInfo("stdin - (cat <file> | %s)", programName) | 		showUsageAndExit() | ||||||
| 			os.Exit(1) | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // showUsageAndExit displays usage information and exits | ||||||
|  | func showUsageAndExit() { | ||||||
|  | 	LogInfo("No input provided") | ||||||
|  | 	LogInfo("Provide input as: ") | ||||||
|  | 	LogInfo("Arguments - %s <source>,<target>,<force?>", programName) | ||||||
|  | 	LogInfo("File - %s -f <file>", programName) | ||||||
|  | 	LogInfo("YAML File - %s -f <file.yaml>", programName) | ||||||
|  | 	LogInfo("Folder (finding sync files in folder recursively) - %s -r <folder>", programName) | ||||||
|  | 	LogInfo("stdin - (cat <file> | %s)", programName) | ||||||
|  | 	os.Exit(1) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // handleStatusErrors processes status channel errors | ||||||
|  | func handleStatusErrors(status chan error) { | ||||||
|  | 	for { | ||||||
|  | 		err, ok := <-status | ||||||
|  | 		if !ok { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 		if err != nil { | ||||||
|  | 			LogError("%v", err) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| 	go func() { | // processInstructions processes all instructions from the channel using parallel workers | ||||||
| 		for { | func processInstructions(instructions chan *LinkInstruction) int32 { | ||||||
| 			err, ok := <-status |  | ||||||
| 			if !ok { |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 			if err != nil { |  | ||||||
| 				LogError("%v", err) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	var instructionsDone int32 = 0 | 	var instructionsDone int32 = 0 | ||||||
| 	var wg sync.WaitGroup |  | ||||||
|  | 	// Collect all instructions first | ||||||
|  | 	var allInstructions []*LinkInstruction | ||||||
| 	for { | 	for { | ||||||
| 		instruction, ok := <-instructions | 		instruction, ok := <-instructions | ||||||
| 		if !ok { | 		if !ok { | ||||||
| 			LogInfo("No more instructions to process") | 			LogInfo("No more instructions to process") | ||||||
| 			break | 			break | ||||||
| 		} | 		} | ||||||
|  | 		allInstructions = append(allInstructions, instruction) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Process instructions in parallel using cyutils.WithWorkers | ||||||
|  | 	// Let the library handle worker count - use 4 workers as a reasonable default | ||||||
|  | 	utils.WithWorkers(4, allInstructions, func(workerID int, _ int, instruction *LinkInstruction) { | ||||||
| 		LogInfo("Processing: %s", instruction.String()) | 		LogInfo("Processing: %s", instruction.String()) | ||||||
| 		status := make(chan error) | 		status := make(chan error) | ||||||
| 		go instruction.RunAsync(status) | 		go instruction.RunAsync(status) | ||||||
| 		wg.Add(1) |  | ||||||
| 		err := <-status | 		err := <-status | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			LogError("Failed processing instruction: %v", err) | 			LogError("Failed processing instruction: %v", err) | ||||||
|  | 		} else { | ||||||
|  | 			atomic.AddInt32(&instructionsDone, 1) | ||||||
| 		} | 		} | ||||||
| 		atomic.AddInt32(&instructionsDone, 1) | 	}) | ||||||
| 		wg.Done() |  | ||||||
| 	} | 	return instructionsDone | ||||||
| 	wg.Wait() |  | ||||||
| 	if instructionsDone == 0 { |  | ||||||
| 		LogInfo("No instructions were processed") |  | ||||||
| 		os.Exit(1) |  | ||||||
| 	} |  | ||||||
| 	LogInfo("All done") |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func IsPipeInput() bool { | func IsPipeInput() bool { | ||||||
| @@ -145,58 +180,49 @@ func ReadFromFilesRecursively(input string, output chan *LinkInstruction, status | |||||||
|  |  | ||||||
| 	files := make(chan string, 128) | 	files := make(chan string, 128) | ||||||
| 	fileStatus := make(chan error) | 	fileStatus := make(chan error) | ||||||
| 	var wg sync.WaitGroup |  | ||||||
| 	go GetSyncFilesRecursively(input, files, fileStatus) | 	go GetSyncFilesRecursively(input, files, fileStatus) | ||||||
| 	go func() { |  | ||||||
| 		wg.Wait() |  | ||||||
| 		close(files) |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		for { |  | ||||||
| 			err, ok := <-fileStatus |  | ||||||
| 			if !ok { |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 			if err != nil { |  | ||||||
| 				LogError("Failed to get sync files recursively: %v", err) |  | ||||||
| 				status <- err |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
|  | 	// Collect all files first | ||||||
|  | 	var syncFiles []string | ||||||
| 	for { | 	for { | ||||||
| 		file, ok := <-files | 		file, ok := <-files | ||||||
| 		if !ok { | 		if !ok { | ||||||
| 			LogInfo("No more files to process") |  | ||||||
| 			break | 			break | ||||||
| 		} | 		} | ||||||
| 		wg.Add(1) | 		syncFiles = append(syncFiles, file) | ||||||
| 		go func() { | 	} | ||||||
| 			defer wg.Done() |  | ||||||
| 			LogInfo(file) |  | ||||||
| 			file = NormalizePath(file, workdir) |  | ||||||
| 			LogInfo("Processing file: %s", FormatPathValue(file)) |  | ||||||
|  |  | ||||||
| 			// This "has" to be done because instructions are resolved in relation to cwd | 	// Check for errors from file search | ||||||
| 			fileDir := FileRegex.FindStringSubmatch(file) | 	for { | ||||||
| 			if fileDir == nil { | 		err, ok := <-fileStatus | ||||||
| 				LogError("Failed to extract directory from %s", FormatSourcePath(file)) | 		if !ok { | ||||||
| 				return | 			break | ||||||
| 			} | 		} | ||||||
| 			LogInfo("Changing directory to %s (for %s)", | 		if err != nil { | ||||||
| 				FormatPathValue(fileDir[1]), | 			LogError("Failed to get sync files recursively: %v", err) | ||||||
| 				FormatPathValue(file)) | 			status <- err | ||||||
| 			err := os.Chdir(fileDir[1]) | 		} | ||||||
| 			if err != nil { | 	} | ||||||
| 				LogError("Failed to change directory to %s: %v", |  | ||||||
| 					FormatSourcePath(fileDir[1]), err) | 	// Process each file | ||||||
| 				return | 	for _, file := range syncFiles { | ||||||
| 			} | 		file = NormalizePath(file, workdir) | ||||||
| 			ReadFromFile(file, output, status, false) | 		LogInfo("Processing file: %s", FormatPathValue(file)) | ||||||
| 			// Don't return directory, stay where we are |  | ||||||
| 			os.Chdir(workdir) | 		// Change to the directory containing the sync file | ||||||
| 		}() | 		fileDir := filepath.Dir(file) | ||||||
|  | 		originalDir, _ := os.Getwd() | ||||||
|  | 		err := os.Chdir(fileDir) | ||||||
|  | 		if err != nil { | ||||||
|  | 			LogError("Failed to change directory to %s: %v", FormatSourcePath(fileDir), err) | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Read and process the file | ||||||
|  | 		ReadFromFile(file, output, status, false) | ||||||
|  |  | ||||||
|  | 		// Return to original directory | ||||||
|  | 		os.Chdir(originalDir) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -212,7 +238,7 @@ func ReadFromFile(input string, output chan *LinkInstruction, status chan error, | |||||||
| 	// Check if this is a YAML file | 	// Check if this is a YAML file | ||||||
| 	if IsYAMLFile(input) { | 	if IsYAMLFile(input) { | ||||||
| 		LogInfo("Parsing as YAML file") | 		LogInfo("Parsing as YAML file") | ||||||
| 		instructions, err := ParseYAMLFile(input, filepath.Dir(input)) | 		instructions, err := ParseYAMLFileRecursive(input, filepath.Dir(input)) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			LogError("Failed to parse YAML file %s: %v", | 			LogError("Failed to parse YAML file %s: %v", | ||||||
| 				FormatSourcePath(input), err) | 				FormatSourcePath(input), err) | ||||||
|   | |||||||
							
								
								
									
										52
									
								
								release.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								release.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | echo "Figuring out the tag..." | ||||||
|  | TAG=$(git describe --tags --exact-match 2>/dev/null || echo "") | ||||||
|  | if [ -z "$TAG" ]; then | ||||||
|  |   # Get the latest tag | ||||||
|  |   LATEST_TAG=$(git describe --tags $(git rev-list --tags --max-count=1)) | ||||||
|  |   # Increment the patch version | ||||||
|  |   IFS='.' read -r -a VERSION_PARTS <<< "$LATEST_TAG" | ||||||
|  |   VERSION_PARTS[2]=$((VERSION_PARTS[2]+1)) | ||||||
|  |   TAG="${VERSION_PARTS[0]}.${VERSION_PARTS[1]}.${VERSION_PARTS[2]}" | ||||||
|  |   # Create a new tag | ||||||
|  |   git tag $TAG | ||||||
|  |   git push origin $TAG | ||||||
|  | fi | ||||||
|  | echo "Tag: $TAG" | ||||||
|  |  | ||||||
|  | echo "Building the thing..." | ||||||
|  | sh build.sh | ||||||
|  | sh install.sh | ||||||
|  |  | ||||||
|  | echo "Creating a release..." | ||||||
|  | TOKEN="$GITEA_API_KEY" | ||||||
|  | GITEA="https://git.site.quack-lab.dev" | ||||||
|  | REPO="dave/synclib" | ||||||
|  | # Create a release | ||||||
|  | RELEASE_RESPONSE=$(curl -s -X POST \ | ||||||
|  |   -H "Authorization: token $TOKEN" \ | ||||||
|  |   -H "Accept: application/json" \ | ||||||
|  |   -H "Content-Type: application/json" \ | ||||||
|  |   -d '{ | ||||||
|  |     "tag_name": "'"$TAG"'", | ||||||
|  |     "name": "'"$TAG"'", | ||||||
|  |     "draft": false, | ||||||
|  |     "prerelease": false | ||||||
|  |   }' \ | ||||||
|  |   $GITEA/api/v1/repos/$REPO/releases) | ||||||
|  |  | ||||||
|  | # Extract the release ID | ||||||
|  | echo $RELEASE_RESPONSE | ||||||
|  | RELEASE_ID=$(echo $RELEASE_RESPONSE | awk -F'"id":' '{print $2+0; exit}') | ||||||
|  | echo "Release ID: $RELEASE_ID" | ||||||
|  |  | ||||||
|  | echo "Uploading the things..." | ||||||
|  | curl -X POST \ | ||||||
|  |   -H "Authorization: token $TOKEN" \ | ||||||
|  |   -F "attachment=@cln.exe" \ | ||||||
|  |   "$GITEA/api/v1/repos/$REPO/releases/${RELEASE_ID}/assets?name=cln.exe" | ||||||
|  | curl -X POST \ | ||||||
|  |   -H "Authorization: token $TOKEN" \ | ||||||
|  |   -F "attachment=@cln" \ | ||||||
|  |   "$GITEA/api/v1/repos/$REPO/releases/${RELEASE_ID}/assets?name=cln" | ||||||
							
								
								
									
										26
									
								
								synclib.log
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								synclib.log
									
									
									
									
									
								
							| @@ -1,26 +0,0 @@ | |||||||
| 18:24:37.967093 Using default sync.yaml file |  | ||||||
| 18:24:37.967093 Input 'sync.yaml' is not absolute, prepending work dir '.' |  | ||||||
| 18:24:37.967593 Reading input from file: [0;32mC:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m |  | ||||||
| 18:24:37.967593 Parsing as YAML file |  | ||||||
| 18:24:37.967593 [4;31mFailed to parse YAML file [0;35mC:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m: [4;31merror unmarshaling YAML: yaml: unmarshal errors: |  | ||||||
|   line 1: cannot unmarshal !!seq into main.YAMLConfig[0m[0m |  | ||||||
| 18:24:37.967593 No more instructions to process |  | ||||||
| 18:24:37.968092 No instructions were processed |  | ||||||
| 18:27:59.691333 Using default sync.yaml file |  | ||||||
| 18:27:59.691333 Input 'sync.yaml' is not absolute, prepending work dir '.' |  | ||||||
| 18:27:59.691834 Reading input from file: [0;32mC:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m |  | ||||||
| 18:27:59.691834 Parsing as YAML file |  | ||||||
| 18:27:59.692335 [0;35mExpanding wildcard source \* in YAML file C:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m |  | ||||||
| 18:27:59.692335 Expanded wildcard source [0;35m\*[0m to 0 links |  | ||||||
| 18:27:59.692836 Expanded wildcard source [0;35m\*[0m in YAML file [0;35mC:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m to 0 links |  | ||||||
| 18:27:59.692836 No more instructions to process |  | ||||||
| 18:27:59.692836 No instructions were processed |  | ||||||
| 18:28:04.075821 Using default sync.yaml file |  | ||||||
| 18:28:04.076320 Input 'sync.yaml' is not absolute, prepending work dir '.' |  | ||||||
| 18:28:04.076320 Reading input from file: [0;32mC:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m |  | ||||||
| 18:28:04.076320 Parsing as YAML file |  | ||||||
| 18:28:04.076320 [0;35mExpanding wildcard source \* in YAML file C:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m |  | ||||||
| 18:28:04.076821 Expanded wildcard source [0;35m\*[0m to 0 links |  | ||||||
| 18:28:04.076821 Expanded wildcard source [0;35m\*[0m in YAML file [0;35mC:/Users/Administrator/Seafile/Projects-Go/GoProjects/synclib/sync.yaml[0m to 0 links |  | ||||||
| 18:28:04.076821 No more instructions to process |  | ||||||
| 18:28:04.076821 No instructions were processed |  | ||||||
							
								
								
									
										117
									
								
								util.go
									
									
									
									
									
								
							
							
						
						
									
										117
									
								
								util.go
									
									
									
									
									
								
							| @@ -5,9 +5,8 @@ import ( | |||||||
| 	"os" | 	"os" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"sync" |  | ||||||
| 	"sync/atomic" | 	"github.com/bmatcuk/doublestar/v4" | ||||||
| 	"time" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func IsSymlink(path string) (bool, error) { | func IsSymlink(path string) (bool, error) { | ||||||
| @@ -76,112 +75,24 @@ func GetSyncFilesRecursively(input string, output chan string, status chan error | |||||||
| 	defer close(output) | 	defer close(output) | ||||||
| 	defer close(status) | 	defer close(status) | ||||||
|  |  | ||||||
| 	var filesProcessed int32 | 	workdir, _ := os.Getwd() | ||||||
| 	var foldersProcessed int32 | 	input = NormalizePath(input, workdir) | ||||||
| 	var activeWorkers int32 | 	LogInfo("Searching for sync files recursively starting in %s", FormatPathValue(input)) | ||||||
|  |  | ||||||
| 	progressTicker := time.NewTicker(200 * time.Millisecond) | 	// Use doublestar to find all sync.yml and sync.yaml files recursively | ||||||
| 	defer progressTicker.Stop() | 	pattern := "**/sync.y*ml" | ||||||
|  | 	files, err := doublestar.Glob(os.DirFS(input), pattern) | ||||||
| 	done := make(chan struct{}) |  | ||||||
| 	defer close(done) |  | ||||||
|  |  | ||||||
| 	directories := make(chan string, 100000) |  | ||||||
| 	workerPool := make(chan struct{}, 4000) |  | ||||||
| 	directories <- input |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		for { |  | ||||||
| 			select { |  | ||||||
| 			case <-progressTicker.C: |  | ||||||
| 				dirCount := len(directories) |  | ||||||
| 				workers := atomic.LoadInt32(&activeWorkers) |  | ||||||
| 				fmt.Printf("\rFiles processed: %d; Folders processed: %d; Active workers: %d; Directory queue: %d", |  | ||||||
| 					atomic.LoadInt32(&filesProcessed), |  | ||||||
| 					atomic.LoadInt32(&foldersProcessed), |  | ||||||
| 					workers, |  | ||||||
| 					dirCount) |  | ||||||
| 			case <-done: |  | ||||||
| 				// Final progress update |  | ||||||
| 				fmt.Printf("\nFiles processed: %d; Folders processed: %d; Completed successfully\n", |  | ||||||
| 					atomic.LoadInt32(&filesProcessed), |  | ||||||
| 					atomic.LoadInt32(&foldersProcessed)) |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	allDone := make(chan struct{}) |  | ||||||
|  |  | ||||||
| 	go func() { |  | ||||||
| 		// WTF is this waitgroup? |  | ||||||
| 		// Nowhere is it added... |  | ||||||
| 		var wg sync.WaitGroup |  | ||||||
|  |  | ||||||
| 		go func() { |  | ||||||
| 			for { |  | ||||||
| 				if atomic.LoadInt32(&activeWorkers) == 0 && len(directories) == 0 { |  | ||||||
| 					time.Sleep(10 * time.Millisecond) |  | ||||||
| 					if atomic.LoadInt32(&activeWorkers) == 0 && len(directories) == 0 { |  | ||||||
| 						close(allDone) |  | ||||||
| 						return |  | ||||||
| 					} |  | ||||||
| 				} |  | ||||||
| 				time.Sleep(50 * time.Millisecond) |  | ||||||
| 			} |  | ||||||
| 		}() |  | ||||||
|  |  | ||||||
| 		for { |  | ||||||
| 			select { |  | ||||||
| 			case directory, ok := <-directories: |  | ||||||
| 				if !ok { |  | ||||||
| 					wg.Wait() |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
|  |  | ||||||
| 				atomic.AddInt32(&activeWorkers, 1) |  | ||||||
|  |  | ||||||
| 				go func(dir string) { |  | ||||||
| 					workerPool <- struct{}{} |  | ||||||
|  |  | ||||||
| 					atomic.AddInt32(&foldersProcessed, 1) |  | ||||||
| 					processDirectory(dir, directories, output, &filesProcessed) |  | ||||||
|  |  | ||||||
| 					<-workerPool |  | ||||||
| 					atomic.AddInt32(&activeWorkers, -1) |  | ||||||
| 				}(directory) |  | ||||||
| 			} |  | ||||||
| 		} |  | ||||||
| 	}() |  | ||||||
|  |  | ||||||
| 	<-allDone |  | ||||||
|  |  | ||||||
| 	if atomic.LoadInt32(&filesProcessed) > 0 { |  | ||||||
| 		LogInfo("Files processed: %d; Folders processed: %d", |  | ||||||
| 			atomic.LoadInt32(&filesProcessed), |  | ||||||
| 			atomic.LoadInt32(&foldersProcessed)) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func processDirectory(directory string, directories chan<- string, output chan<- string, filesProcessed *int32) { |  | ||||||
| 	files, err := os.ReadDir(directory) |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		LogError("Error reading directory %s: %v", directory, err) | 		LogError("Failed to search for pattern %s: %v", pattern, err) | ||||||
|  | 		status <- err | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, file := range files { | 	for _, file := range files { | ||||||
| 		if file.IsDir() { | 		fullPath := filepath.Join(input, file) | ||||||
| 			directories <- filepath.Join(directory, file.Name()) | 		LogInfo("Found sync file: %s", FormatPathValue(fullPath)) | ||||||
| 		} else { | 		output <- fullPath | ||||||
| 			if IsYAMLSyncFile(file.Name()) { |  | ||||||
| 				output <- filepath.Join(directory, file.Name()) |  | ||||||
| 			} |  | ||||||
| 			atomic.AddInt32(filesProcessed, 1) |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| } |  | ||||||
|  |  | ||||||
| func IsYAMLSyncFile(filename string) bool { | 	LogInfo("Completed recursive search for sync files") | ||||||
| 	return filename == "sync.yaml" || filename == "sync.yml" |  | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user