-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathscanRun.go
160 lines (137 loc) · 3.59 KB
/
scanRun.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
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
102
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
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
package m3ugen
import (
"bufio"
"fmt"
"log"
"os"
"regexp"
"strings"
)
const (
initialFoundFilesPathCapacity = 1 * (1024 ^ 2) // 1 Mi
)
// ScanRun represents a scan & playlist generation process.
type ScanRun struct {
Config *Config
FoundFilesPaths []string
// FoundExtensions is a list of observed extensions. Value is true when
// the extension was considered and false when excluded.
FoundExtensions map[string]bool
verbose func(f string, args ...any)
debug func(f string, args ...any)
}
var (
regexGetFileExtension = regexp.MustCompile(`^.*\.(.*)$`)
)
// Start begins the process of scanning and generating the playlist.
func Start(config *Config) (*ScanRun, error) {
if err := config.Validate(); err != nil {
return nil, err
}
r := &ScanRun{
Config: config,
FoundFilesPaths: make([]string, 0, initialFoundFilesPathCapacity),
}
r.initializeVerboseAndDebugOutputs()
r.debug("Starting scan & generate process using config %+v", config)
if err := r.scan(); err != nil {
return nil, err
}
if config.DetectDuplicates {
r.detectDuplicates()
}
if err := r.writePlaylist(); err != nil {
return nil, err
}
r.logExcludedExtensions()
return r, nil
}
func (r *ScanRun) initializeVerboseAndDebugOutputs() {
// VERBOSE (implicit if 'Debug' activated)
if r.Config.Verbose || r.Config.Debug {
r.verbose = func(format string, a ...any) {
fmt.Fprintln(os.Stderr, fmt.Sprintf(format, a...))
}
} else {
r.verbose = func(format string, a ...any) {}
}
// DEBUG
if r.Config.Debug {
r.debug = func(format string, a ...any) {
line := fmt.Sprintf(format, a...)
log.Default().Println(line)
}
} else {
r.debug = func(format string, a ...any) {}
}
}
func (r *ScanRun) writePlaylist() (err error) {
fileList := make([]string, len(r.FoundFilesPaths))
copy(fileList, r.FoundFilesPaths)
if r.Config.RandomizeList {
r.verbose("Shuffling the found files")
ShuffleSlice(fileList)
}
foundFilesPathsCount := len(r.FoundFilesPaths)
max := r.Config.MaximumEntries
if max < 1 {
r.verbose("No maximum entries. Writing all %d files to output.", foundFilesPathsCount)
max = foundFilesPathsCount
} else if max > foundFilesPathsCount {
r.verbose("Limited to %d. Writing all %d found files to output.", max, foundFilesPathsCount)
max = foundFilesPathsCount
} else {
r.verbose("Limited to %d. Writing the first %d found files to output.", max, max)
}
r.verbose("Writing playlist to %s", r.Config.OutputPath)
f, err := os.OpenFile(r.Config.OutputPath, os.O_CREATE|os.O_TRUNC|os.O_RDWR, 0644)
if err != nil {
return
}
defer func() {
err = FirstErr(err, f.Close())
}()
w := bufio.NewWriter(f)
defer func() {
err = FirstErr(err, w.Flush())
}()
for _, entry := range fileList[:max] {
_, err = fmt.Fprintln(w, entry)
if err != nil {
return
}
}
return
}
func (r *ScanRun) logExcludedExtensions() {
if !r.Config.Verbose {
return
}
excluded := make([]string, 0, len(r.FoundExtensions))
for extension, included := range r.FoundExtensions {
if !included {
excluded = append(excluded, extension)
}
}
excludedList := strings.Join(excluded, ", ")
r.verbose("Extensions not considered: %s", excludedList)
}
func (r *ScanRun) detectDuplicates() {
r.verbose("Detecting duplicates")
fileCounter := make(map[string]int)
for _, f := range r.FoundFilesPaths {
c, ok := fileCounter[f]
if !ok {
c = 0
}
fileCounter[f] = c + 1
}
duplicatesCount := 0
for f, c := range fileCounter {
if c > 1 {
r.verbose("File %q is present %d times in the search", f, c)
duplicatesCount++
}
}
r.verbose("%d files were detected as duplicates", duplicatesCount)
}