forked from dakyskye/dxhd
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathparser.go
634 lines (561 loc) · 17.1 KB
/
parser.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
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
package main
import (
"bufio"
"errors"
"fmt"
"io"
"os"
"regexp"
"strconv"
"strings"
"go.uber.org/zap"
)
// event definitions
const (
evtKeyPress int = iota
evtKeyRelease
evtButtonPress
evtButtonRelease
)
// filedata holds the data of parsed file
type filedata struct {
originalBinding string
binding strings.Builder
action strings.Builder
evtType int
hasVariant bool
}
// ranges hold the data of a keybinding and it's action
type ranges struct {
binding rng
action struct {
rng
skip bool
numerical bool
}
}
type rng struct {
start, end int
startStr, endStr string
}
type variantGroup struct {
action, binding []string
}
// global regular expressions, compiled once at run-time
var (
keybindingPattern = regexp.MustCompile(`^#(((!?@?)|@?!?)\w+{.*?}|((!?@?)|@?!?){.*?}|((!?@?)|@?!?)\w+)(((\+(((!?@?)|@?!?)\w+{.*?}|((!?@?)|@?!?){.*?}|((!?@?)|@?!?)\w+)))+)?`)
variantPattern = regexp.MustCompile(`{.*?}`)
bindingRangePattern = regexp.MustCompile(`([0-9]|[a-z])-([0-9]|[a-z])`)
actionRangePattern = regexp.MustCompile(`(?m)^(([0-9]+)-([0-9]+))|(([a-z])-([a-z]))$`)
numericalPattern = regexp.MustCompile(`([0-9]+)-([0-9]+)`)
alphabeticalPattern = regexp.MustCompile(`([a-z])-([a-z])`)
mouseBindPattern = regexp.MustCompile(`mouse([0-9]+)`)
xfKeyPattern = regexp.MustCompile(`XF86\w+`)
)
// parese function parses a config file, and returns data
func parse(file string, data *[]filedata) (shell string, err error) {
if data == nil {
return "", errors.New("empty value was passed to parse function")
}
configFile, err := os.Open(file)
if err != nil {
return
}
defer func() {
e := configFile.Close()
if e != nil {
if err == nil {
err = e
} else {
zap.L().Debug("failed to close config file", zap.Error(e))
}
}
}()
reader := bufio.NewReader(configFile)
lineNumber := 0
shell = "/bin/sh"
wasKeybinding := false
wasPrefix := false
datum := []filedata{}
index := 0
// read file line by line
for {
lineNumber++
var (
line []byte
isPrefix bool
)
line, isPrefix, err = reader.ReadLine()
if err != nil {
break
}
if index+1 != len(datum) {
datum = append(datum, filedata{})
}
lineStr := string(line)
// skip the shebang
if lineNumber == 1 && strings.HasPrefix(lineStr, "#!") {
shell = lineStr[2:]
continue
}
// skip an empty line
if lineStr == "" {
continue
}
// ignore comments (##+)
if strings.HasPrefix(lineStr, "##") {
continue
}
// decide whether the line is a keybinding or not
if strings.HasPrefix(lineStr, "#") {
if isPrefix {
err = errors.New(fmt.Sprintf("a keybinding can't be that long, line %d, file %s", lineNumber, file))
return
}
// erase spaces for key validation
lineStr = strings.ReplaceAll(lineStr, " ", "")
if keybindingPattern.MatchString(lineStr) {
if datum[index].action.Len() != 0 {
index++
datum = append(datum, filedata{})
}
// trim # prefix
lineStr := lineStr[1:]
// overwrite previous prefix if needed
if wasKeybinding {
if datum[index].binding.Len() != 0 {
datum[index].binding.Reset()
zap.L().Info("overwriting older keybinding", zap.String("file", file), zap.Int("line", lineNumber))
zap.L().Debug("overwriting keybinding", zap.String("old", datum[index].binding.String()), zap.String("new", lineStr))
}
}
// getEventType merges two events into one type
getEventType := func(old, new int) (evt int) {
switch old {
case evtKeyPress:
evt = new
case evtKeyRelease:
if new != evtKeyPress {
evt = new
}
case evtButtonPress:
if new == evtKeyRelease || new == evtButtonRelease {
evt = evtButtonRelease
}
case evtButtonRelease:
evt = evtButtonRelease
default:
evt = new
}
return
}
// set to -1, in case a keybinding is a single letter
datum[index].evtType = -1
for _, key := range strings.Split(lineStr, "+") {
if len(key) > 1 {
evt := -1
if strings.HasPrefix(key, "@mouse") {
evt = evtButtonRelease
} else if strings.HasPrefix(key, "mouse") {
evt = evtButtonPress
} else if strings.HasPrefix(key, "@") {
evt = evtKeyRelease
} else {
evt = evtKeyPress
}
datum[index].evtType = getEventType(datum[index].evtType, evt)
}
}
// means a keybinding was the single letter
if datum[index].evtType == -1 {
datum[index].evtType = evtKeyPress
}
_, err = datum[index].binding.WriteString(lineStr)
if err != nil {
return
}
datum[index].hasVariant = len(variantPattern.FindStringIndex(lineStr)) > 0
wasKeybinding = true
}
} else {
wasKeybinding = false
if isPrefix {
if wasPrefix {
datum[index].action.Write(line)
} else {
if datum[index].action.Len() != 0 {
datum[index].action.Write([]byte("\n"))
}
datum[index].action.Write(line)
wasPrefix = true
}
continue
}
if wasPrefix {
datum[index].action.Write(line)
wasPrefix = false
} else {
if datum[index].action.Len() != 0 {
datum[index].action.Write([]byte("\n"))
}
datum[index].action.Write(line)
}
}
}
// we are ok when we reach the end of line
if err == io.EOF {
err = nil
} else {
return
}
// xgb requires these shorthands to be replaced to what they are called internally
replaceShorthands := func(data *filedata) (err error) {
data.originalBinding = data.binding.String()
data.binding.Reset()
modified := data.originalBinding
// extract xf86 keys if any
matches := xfKeyPattern.FindAllString(modified, -1)
indexes := xfKeyPattern.FindAllStringIndex(modified, -1)
if len(matches) != len(indexes) {
err = errors.New("can not process XF86 keys properly")
return
}
// lowercase whole line
modified = strings.ToLower(modified)
// put XF86 keys as they were before in lineStr
for in, index := range indexes {
modified = strings.Replace(modified, modified[index[0]:index[1]], matches[in], 1)
}
modified = strings.ReplaceAll(data.originalBinding, "+", "-")
modified = strings.ReplaceAll(modified, "super", "mod4")
modified = strings.ReplaceAll(modified, "alt", "mod1")
modified = strings.ReplaceAll(modified, "ctrl", "control")
modified = strings.ReplaceAll(strings.ReplaceAll(modified, "@", ""), "!", "")
// replace mouseN with N
if data.evtType == evtButtonPress || data.evtType == evtButtonRelease {
modified = mouseBindPattern.ReplaceAllString(modified, "$1")
}
_, err = data.binding.WriteString(modified)
return
}
*data = nil
for _, d := range datum {
// replicate a keybinding and it's action if it has variants
if d.hasVariant {
replicated, e := replicate(d.binding.String(), d.action.String())
if e != nil {
err = errors.New(fmt.Sprintf("can't register %s keybinding, error (%s)", strings.TrimPrefix(d.binding.String(), "#"), e.Error()))
return
}
for _, repl := range replicated {
repl.evtType = d.evtType
err = replaceShorthands(repl)
if err != nil {
return
}
*data = append(*data, filedata{originalBinding: repl.originalBinding, binding: repl.binding, action: repl.action, evtType: d.evtType})
}
} else {
err = replaceShorthands(&d)
if err != nil {
return
}
*data = append(*data, filedata{originalBinding: d.originalBinding, binding: d.binding, action: d.action, evtType: d.evtType})
}
}
// means config file was empty
if len(*data) == 1 && ((*data)[0].action.String() == "" || (*data)[0].binding.String() == "") {
err = errors.New("config file does not contain any binding/action")
return
}
return
}
// replicate replicates variants
func replicate(binding, action string) (replicated []*filedata, err error) {
// find all the variants
bindingVariants, actionVariants := variantPattern.FindAllString(binding, -1), variantPattern.FindAllString(action, -1)
// make sure the amount of variants do match
if len(bindingVariants) != len(actionVariants) {
err = errors.New("the amount of variants in a keybinding and it's action do not match")
return
}
var bindingVars, actionVars [][]string
// extract variant members
extract := func(from []string, where *[][]string) {
for _, f := range from {
*where = append(*where, strings.Split(strings.TrimSuffix(strings.TrimPrefix(f, "{"), "}"), ","))
}
}
extract(bindingVariants, &bindingVars)
extract(actionVariants, &actionVars)
// validate the amount of variant memebers do match
for i, b := range bindingVars {
if len(b) != len(actionVars[i]) {
err = errors.New("the amount of variant members in a keybinding and it's action do not match")
return
}
}
// validate and extract ranges
var rngs []ranges
rngs, err = extractRanges(bindingVars, actionVars)
if err != nil {
return
}
var expandedBindingRanges, expandedActionRanges []string
// expands a range in a keybinding({1-9} -> {1},{2},{3},{...},{9})
expandRange := func(r ranges, binding, acton string, bindings, actions *[]string) {
// bindings
for bIn := r.binding.start; bIn != r.binding.end+1; bIn++ {
*bindings = append(*bindings, strings.Replace(
binding,
fmt.Sprintf("%s-%s", r.binding.startStr, r.binding.endStr),
fmt.Sprintf("%c", rune(bIn)),
1,
))
if r.action.skip {
*actions = append(*actions, acton)
}
}
// actions
if !r.action.skip {
for aIn := r.action.start; aIn != r.action.end+1; aIn++ {
if r.action.numerical {
*actions = append(*actions, strings.Replace(
action,
fmt.Sprintf("%s-%s", r.action.startStr, r.action.endStr),
fmt.Sprintf("%d", aIn),
1,
))
} else {
*actions = append(*actions, strings.Replace(
action,
fmt.Sprintf("%s-%s", r.action.startStr, r.action.endStr),
fmt.Sprintf("%c", rune(aIn)),
1,
))
}
}
}
}
// for as long as we have unexpanded ranges, expand them
for len(rngs) > 0 {
if len(expandedBindingRanges) > 0 {
var newBindingRanges, newActionRanges []string
if len(expandedActionRanges) != len(expandedBindingRanges) {
err = errors.New("an unknown error occurred whilst expanding keybinding and action ranges")
}
for i := 0; i != len(expandedBindingRanges); i++ {
expandRange(rngs[0], expandedBindingRanges[i], expandedActionRanges[i], &newBindingRanges, &newActionRanges)
}
expandedBindingRanges, expandedActionRanges = newBindingRanges, newActionRanges
} else {
expandRange(rngs[0], binding, action, &expandedBindingRanges, &expandedActionRanges)
}
rngs = rngs[1:]
}
if len(expandedActionRanges) != len(expandedBindingRanges) {
err = errors.New("an unknown error occurred whilst expanding keybinding and action ranges")
}
// replicateVariant replaces pattern with each member of variants group
replicateVariant := func(in, pattern string, variants []string, where *[]string) {
for _, v := range variants {
if v == "_" {
*where = append(*where, strings.Replace(
in,
pattern,
"",
1,
))
} else {
*where = append(*where, strings.Replace(
in,
pattern,
v,
1,
))
}
}
}
// in case our keybinding and action had no ranges
if len(expandedBindingRanges) == 0 {
expandedBindingRanges = append(expandedBindingRanges, binding)
expandedActionRanges = append(expandedActionRanges, action)
}
// do replicate every variant member
for i, r := 0, 0; i != len(expandedBindingRanges); i++ {
var replicatedBindings, replicatedActions []string
vGroup := &variantGroup{}
vGroup.action = variantPattern.FindAllString(expandedActionRanges[i], -1)
vGroup.binding = variantPattern.FindAllString(expandedBindingRanges[i], -1)
if !(len(vGroup.action) == len(vGroup.binding) && len(vGroup.action) > 0) {
err = errors.New("can not extract variant groups")
return
}
// for as long as we have binding AND action in a variant group
for len(vGroup.binding) > 0 {
// extract variant members
bVariantMembers := strings.Split(strings.TrimSuffix(strings.TrimPrefix(vGroup.binding[0], "{"), "}"), ",")
aVariantMembers := strings.Split(strings.TrimSuffix(strings.TrimPrefix(vGroup.action[0], "{"), "}"), ",")
// if we already replicated a variant, use it
if len(replicatedBindings) > 0 {
var newBindingVariants, newActionVariants []string
for _, alreadyR := range replicatedBindings {
replicateVariant(alreadyR, vGroup.binding[0], bVariantMembers, &newBindingVariants)
}
for _, alreadyR := range replicatedActions {
replicateVariant(alreadyR, vGroup.action[0], aVariantMembers, &newActionVariants)
}
replicatedBindings, replicatedActions = newBindingVariants, newActionVariants
} else {
replicateVariant(expandedBindingRanges[i], vGroup.binding[0], bVariantMembers, &replicatedBindings)
replicateVariant(expandedActionRanges[i], vGroup.action[0], aVariantMembers, &replicatedActions)
}
vGroup.binding = vGroup.binding[1:]
vGroup.action = vGroup.action[1:]
}
if len(replicatedBindings) != len(replicatedActions) {
err = errors.New("replication went wrong")
return
}
// append replicated bindings and actions to the return result
appender:
for i := 0; i != len(replicatedBindings); i++ {
// we get ++ when we replace underscore literal with nothing
replicatedBindings[i] = strings.ReplaceAll(replicatedBindings[i], "++", "+")
if i > 0 {
for _, aR := range replicated {
if aR.binding.String() == replicatedBindings[i] {
continue appender
}
}
}
replicated = append(replicated, &filedata{})
_, err = replicated[r].binding.WriteString(replicatedBindings[i])
if err != nil {
return
}
_, err = replicated[r].action.WriteString(replicatedActions[i])
if err != nil {
return
}
r++
}
}
return
}
// extracts every range from a config file
func extractRanges(bindingVars, actionVars [][]string) (r []ranges, err error) {
// range patterns for binding and action and range errors
var (
rangeParseErr = errors.New("could not parse a range")
invalidRangeErr = errors.New("invalid parse given")
)
// iterate over a binding variants, and replicate ranges
for bIn, bVars := range bindingVars {
// iterate over the variants of the binding variants
for vIn, bVar := range bVars {
// check if the variant is a range
if bindingRangePattern.MatchString(bVar) {
bRange := bindingRangePattern.FindStringSubmatch(bVar)
if len(bRange) != 3 {
err = rangeParseErr
return
}
var (
aVar = actionVars[bIn][vIn]
aRange []string
aRangeValidation = true
)
// make sure action variant is also a range (or _)
if !actionRangePattern.MatchString(aVar) {
if aVar == "_" {
// in case it's _, skip the range validation
aRangeValidation = false
} else {
err = errors.New("the indexes of ranges for a keybinding and it's action do not match")
return
}
}
// int values for comparison
// string values for assignment
var (
bStart, bEnd int
bStartStr, bEndStr string
)
// convert strings to runes to int
bStart = int([]rune(bRange[1])[0])
bEnd = int([]rune(bRange[2])[0])
// make sure the given range is valid
if bStart >= bEnd {
err = invalidRangeErr
return
}
// assign string values
bStartStr = string(rune(bStart))
bEndStr = string(rune(bEnd))
var (
aStart, aEnd int
aStartStr, aEndStr string
aNumerical = false
)
if aRangeValidation {
if numericalPattern.MatchString(aVar) { // is it a numerical range?
aRange = numericalPattern.FindStringSubmatch(aVar)
if len(aRange) != 3 {
err = rangeParseErr
return
}
aStart, err = strconv.Atoi(aRange[1])
if err != nil {
return
}
aEnd, err = strconv.Atoi(aRange[2])
if err != nil {
return
}
for _, a := range []rune(aRange[1]) {
aStartStr += string(a)
}
for _, a := range []rune(aRange[2]) {
aEndStr += string(a)
}
aNumerical = true
} else if alphabeticalPattern.MatchString(aVar) { // well, is it an alphabetical range?
aRange = alphabeticalPattern.FindStringSubmatch(aVar)
if len(aRange) != 3 {
err = rangeParseErr
return
}
aStart = int([]rune(aRange[1])[0])
aEnd = int([]rune(aRange[2])[0])
aStartStr = string(rune(aStart))
aEndStr = string(rune(aEnd))
} else { // it's an invalid range
err = invalidRangeErr
return
}
// make sure the ranges match
// 1-9 compared to 11-19 is a valid range
// 4-8 compared to a-e is a valid range also
// b start-end a start-end
if (bStart - bEnd) != (aStart - aEnd) {
err = errors.New("the ranges of a keybinding and it's action do not match")
return
}
}
r = append(r, ranges{})
r[len(r)-1].binding.start = bStart
r[len(r)-1].binding.startStr = bStartStr
r[len(r)-1].binding.end = bEnd
r[len(r)-1].binding.endStr = bEndStr
r[len(r)-1].action.start = aStart
r[len(r)-1].action.startStr = aStartStr
r[len(r)-1].action.end = aEnd
r[len(r)-1].action.endStr = aEndStr
r[len(r)-1].action.skip = aRangeValidation == false
r[len(r)-1].action.numerical = aNumerical
}
}
}
return
}