-
-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathcmd_comments.go
217 lines (176 loc) · 5.91 KB
/
cmd_comments.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
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
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
package main
import (
"flag"
"fmt"
"os"
"regexp"
"sort"
"strings"
)
// Comment is a structure to hold a language/syntax for comments.
//
// A comment is denoted as the content between a start-marker, and an
// end-marker. For single-line comments we define the end-marker as
// being a newline.
type Comment struct {
// The text which denotes the start of a comment.
//
// For C++ this might be `/*`, for a shell-script it might be `#`.
start string
// The text which denotes the end of a comment.
//
// For C++ this might be `*/`, for a shell-script it might be `\n`.
end string
// Some comment-openers are only valid at the start of a line.
bol bool
}
// Structure for our options and state.
type commentsCommand struct {
// The styles of comments to be enabled, as set by the command-line.
style string
// Pretty-print the comments?
pretty bool
// The comments we're matching
patterns []Comment
}
// Arguments adds per-command args to the object.
func (cc *commentsCommand) Arguments(f *flag.FlagSet) {
f.StringVar(&cc.style, "style", "c,cpp", "A comma-separated list of the comment-styles to use")
f.BoolVar(&cc.pretty, "pretty", false, "Reformat comments for readability")
}
// Info returns the name of this subcommand.
func (cc *commentsCommand) Info() (string, string) {
return "comments", `Output the comments contained in the given file.
Details:
This naive command outputs the comments which are included in the specified
filename(s). This is useful if you wish to run spell-checkers, etc.
There is support for outputting single-line and multi-line comments for C,
C++, Lua, and Golang. Additional options are welcome. By default C, and
C++ are enabled. To only use Lua comments you could run:
$ sysbox comments --style=lua *.lua`
}
// showComment writes the comment to the console, after optionally tidying
func (cc *commentsCommand) showComment(comment string) {
if cc.pretty {
// Remove newlines
comment = strings.Replace(comment, "\n", " ", -1)
// Remove " * "
comment = strings.Replace(comment, " * ", " ", -1)
// Collapse adjacent spaces
comment = strings.Join(strings.Fields(comment), " ")
// Skip empty comments; i.e. just literal matches of
// the opening pattern.
for _, pattern := range cc.patterns {
if comment == pattern.start {
return
}
}
}
// Remove trailing newline, so we can safely add one
comment = strings.TrimSuffix(comment, "\n")
fmt.Printf("%s\n", comment)
}
// dumpComments dumps the comments from the given file.
func (cc *commentsCommand) dumpComments(filename string) {
// Read the content
content, err := os.ReadFile(filename)
if err != nil {
fmt.Printf("error reading %s: %s\n", filename, err.Error())
return
}
// Convert our internal patterns to a series of regular expressions.
var r []*regexp.Regexp
for _, pattern := range cc.patterns {
reg := "(?s)"
if pattern.bol {
reg += "^"
}
reg += regexp.QuoteMeta(pattern.start)
reg += "(.*?)"
reg += regexp.QuoteMeta(pattern.end)
fmt.Printf("%v\n", reg)
r = append(r, regexp.MustCompile(reg))
}
// Now for each regexp do the matching over the whole input.
for _, re := range r {
out := re.FindAllSubmatch(content, -1)
for _, match := range out {
cc.showComment(string(match[0]))
}
}
}
// Execute is invoked if the user specifies `comments` as the subcommand.
func (cc *commentsCommand) Execute(args []string) int {
// Map of known patterns, by name
known := make(map[string][]Comment)
// Populate with the patterns.
known["ada"] = []Comment{{start: "--", end: "\n"}}
known["apl"] = []Comment{{start: "⍝", end: "\n"}}
known["applescript"] = []Comment{{start: "(*", end: "*)"},
{start: "--", end: "\n"}}
known["asm"] = []Comment{{start: ";", end: "\n"}}
known["basic"] = []Comment{{start: "REM", end: "\n"}}
known["c"] = []Comment{{start: "//", end: "\n"}}
known["coldfusion"] = []Comment{{start: "<!---", end: "--->"}}
known["cpp"] = []Comment{{start: "/*", end: "*/"}}
known["fortran"] = []Comment{{start: "!", end: "\n", bol: true}}
known["go"] = []Comment{{start: "/*", end: "*/"},
{start: "//", end: "\n"},
}
known["html"] = []Comment{{start: "<!--", end: "-->"}}
known["haskell"] = []Comment{{start: "{-", end: "-}"},
{start: "--", end: "\n"}}
known["lisp"] = []Comment{{start: ";", end: "\n"}}
known["java"] = []Comment{{start: "/*", end: "*/"},
{start: "//", end: "\n"}}
known["javascript"] = []Comment{{start: "/*", end: "*/"},
{start: "//", end: "\n"}}
known["lua"] = []Comment{{start: "--[[", end: "--]]"},
{start: "-- ", end: "\n"}}
known["matlab"] = []Comment{{start: "%{", end: "%}"},
{start: "% ", end: "\n"}}
known["pascal"] = []Comment{{start: "(*", end: "*)"}}
known["perl"] = []Comment{{start: "#", end: "\n"}}
known["php"] = []Comment{{start: "/*", end: "*/"},
{start: "//", end: "\n"},
{start: "#", end: "\n"},
}
known["python"] = []Comment{{start: "#", end: "\n"}}
known["ruby"] = []Comment{{start: "#", end: "\n"}}
known["shell"] = []Comment{{start: "#", end: "\n"}}
known["swift"] = []Comment{{start: "/*", end: "*/"},
{start: "//", end: "\n"}}
known["sql"] = []Comment{{start: "--", end: "\n"}}
known["xml"] = []Comment{{start: "<!--", end: "-->"}}
// Ensure we have at least one filename specified.
if len(args) <= 0 {
fmt.Printf("Usage: comments file1 [file2] ..[argN]\n")
return 1
}
// Load the patterns the user selected.
for _, kind := range strings.Split(cc.style, ",") {
// Lookup the choice
pat, ok := known[kind]
// Not found? That's an error
if !ok {
fmt.Printf("Unknown style %s, valid options include:\n", kind)
keys := make([]string, 0, len(known))
for k := range known {
keys = append(keys, k)
}
sort.Strings(keys)
for _, k := range keys {
fmt.Printf("\t%s\n", k)
}
return 1
}
// Otherwise add it to the list.
cc.patterns = append(cc.patterns, pat...)
}
// Now process the input file(s)
for _, file := range args {
cc.dumpComments(file)
}
// All done.
return 0
}