-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathstrings.go
605 lines (556 loc) · 17.2 KB
/
strings.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
// Copyright 2021 Josh Deprez
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package yarn
import (
"encoding/csv"
"fmt"
"io"
"io/fs"
"os"
"sort"
"strconv"
"strings"
"github.com/alecthomas/participle/v2"
"github.com/alecthomas/participle/v2/lexer"
cldr "github.com/razor-1/localizer-cldr"
"golang.org/x/text/feature/plural"
"golang.org/x/text/language"
)
// StringTable contains all the information from a string table, keyed by
// string ID. This can be constructed either by using ReadStringTable, or
// manually (e.g. if you are not using Yarn Spinner CSV string tables but still
// want to use substitutions, format functions, and markup tags).
type StringTable struct {
Language language.Tag
Table map[string]*StringTableRow
}
// LoadStringTableFile is a convenient function for loading a CSV string table
// given a file path. If stringTablePath is foo/bar/file-Lines.csv then it expects
// a corresponding Metadata file at foo/bar/file-Metadata.csv. It assumes the first
// row of both files are a header. langCode must be a valid BCP 47 language tag.
func LoadStringTableFile(stringTablePath, langCode string) (*StringTable, error) {
csv, err := os.Open(stringTablePath)
if err != nil {
return nil, fmt.Errorf("opening string table file: %w", err)
}
defer csv.Close()
st, err := ReadStringTable(csv, langCode)
if err != nil {
return nil, fmt.Errorf("reading string table: %w", err)
}
csv, err = os.Open(metadataTablePath(stringTablePath))
if err != nil {
return nil, fmt.Errorf("opening metadata file: %w", err)
}
if err := st.readMetadata(csv); err != nil {
return nil, fmt.Errorf("reading metadata file: %w", err)
}
return st, nil
}
// LoadStringTableFileFS loads compiled Yarn Spinner files from the provided fs.FS.
// See LoadStringTableFile for details.
func LoadStringTableFileFS(fsys fs.FS, stringTablePath, langCode string) (*StringTable, error) {
csv, err := fsys.Open(stringTablePath)
if err != nil {
return nil, fmt.Errorf("opening string table file: %w", err)
}
defer csv.Close()
st, err := ReadStringTable(csv, langCode)
if err != nil {
return nil, fmt.Errorf("reading string table: %w", err)
}
csv, err = fsys.Open(metadataTablePath(stringTablePath))
if err != nil {
return nil, fmt.Errorf("opening metadata file: %w", err)
}
if err := st.readMetadata(csv); err != nil {
return nil, fmt.Errorf("reading metadata table: %w", err)
}
return st, nil
}
// ReadStringTable reads a CSV string table from the reader. It assumes the
// first row is a header. langCode must be a valid BCP 47 language tag.
// In addition to checking the CSV structure as it is parsed, each lineNumber
// is parsed as an int, and each text is also parsed. Any malformed substitution
// tokens or markup tags will cause an error.
func ReadStringTable(r io.Reader, langCode string) (*StringTable, error) {
lang, err := language.Parse(langCode)
if err != nil {
return nil, fmt.Errorf("invalid lang code: %w", err)
}
st := make(map[string]*StringTableRow)
header := true
cr := csv.NewReader(r)
cr.FieldsPerRecord = 5
for {
rec, err := cr.Read()
if err == io.EOF {
break
}
if err != nil {
return nil, fmt.Errorf("csv read: %w", err)
}
if header {
header = false
continue
}
// Line number must be an int
ln, err := strconv.Atoi(rec[4])
if err != nil {
return nil, fmt.Errorf("line number not an int: %w", err)
}
id := rec[0]
row := &StringTableRow{
ID: id,
Text: rec[1],
File: rec[2],
Node: rec[3],
LineNumber: ln,
}
// Text must be parseable - parse it now to catch errors sooner
if err := row.parseIfNeeded(); err != nil {
return nil, fmt.Errorf("text for id %s could not be parsed: %w", id, err)
}
st[id] = row
}
return &StringTable{
Language: lang,
Table: st,
}, nil
}
// readMetadata extracts tags from the metadata table.
func (t *StringTable) readMetadata(r io.Reader) error {
header := true
cr := csv.NewReader(r)
cr.FieldsPerRecord = -1 // tags can be multirow
for {
rec, err := cr.Read()
if err == io.EOF {
break
}
if err != nil {
return fmt.Errorf("csv error: %w", err)
}
if header {
header = false
continue
}
if len(rec) < 4 { // nothing to do if there are fewer than 4 rows.
continue
}
id := rec[0]
row, ok := t.Table[id]
if !ok {
return fmt.Errorf("unexpected ID in metadata table: %q", id)
}
row.Tags = rec[3:]
}
return nil
}
// Render looks up the row corresponding to line.ID, interpolates substitutions
// (from line.Substitutions), applies format functions, and processes style
// tags into attributes.
func (t *StringTable) Render(line Line) (*AttributedString, error) {
row := t.Table[line.ID]
if row == nil {
return nil, fmt.Errorf("string table row for id %q not found or nil", line.ID)
}
return row.Render(line.Substitutions, t.Language)
}
// StringTableRow contains all the information from one row in a string table.
type StringTableRow struct {
ID, Text, File, Node string
LineNumber int
origText string // parsedText needs updating if Text changes
parsedText *parsedString
Tags []string // Tags are set in the metadata table.
}
// Render interpolates substitutions, applies format functions, and processes
// style tags into attributes.
func (r *StringTableRow) Render(substs []string, lang language.Tag) (*AttributedString, error) {
if err := r.parseIfNeeded(); err != nil {
return nil, err
}
lr := lineRenderer{
substs: substs,
lang: lang,
}
if err := lr.renderString(r.parsedText); err != nil {
return nil, err
}
return lr.attStr(), nil
}
// parseIfNeeded parses r.Text, if it has not been parsed already.
func (r *StringTableRow) parseIfNeeded() error {
if r.Text == r.origText && r.parsedText != nil {
return nil
}
filename := fmt.Sprintf("%s:%d", r.File, r.LineNumber)
pt, err := lineParser.ParseString(filename, r.Text)
if err != nil {
return err
}
r.origText = r.Text
r.parsedText = pt
return nil
}
// AttributedString is a string with additional attributes, such as presentation
// or styling information, that apply to the whole string or substrings.
type AttributedString struct {
str string
atts map[int][]*Attribute // position -> attributes starting or ending here
}
func (s *AttributedString) String() string { return s.str }
// ScanAttribEvents calls visit with each change in attribute state. pos is the
// byte position in the string where the change occurs. atts will contain the
// attributes that either start or end at pos, in the same order they were read
// from the original markup. Self-closing tags, or an open and close pair that
// apply to the same position (i.e. marking up nothing) will only be present in
// atts once (in the order of the start tag).
// For example, for the original string:
//
// `[a]Hello A[/a] [b]Hello B[/b] [c][d][/c]No C, [e/]only D[/d]`
//
// which is processed into the unattributed string:
//
// `Hello A Hello B No C, only D`
//
// ScanAttribEvents will visit:
// * (0, [a]) -- open of a
// * (7, [a]) -- close of a
// * (8, [b]) -- open of b
// * (15, [b]) -- close of b
// * (16, [c,d]) -- close of c applies to same position, so it appears once
// * (22, [e]) -- e is self-closing, so it appears once
// * (28, [d]) -- close of d
func (s *AttributedString) ScanAttribEvents(visit func(pos int, atts []*Attribute)) {
events := make([]int, 0, len(s.atts))
for i := range s.atts {
events = append(events, i)
}
sort.Ints(events)
for _, pos := range events {
visit(pos, s.atts[pos])
}
}
// Attribute describes a range within a string with additional information
// provided by markup tags. Start and End specify the range in bytes. Name is
// the tag name, and Props contains any additional key="value" tag properties.
type Attribute struct {
Start, End int
Name string
Props map[string]string
}
var (
// This lexer is a bit more general than needed since it allows things like
// nested functions, but... hey I get nested functions for ~free!
lineLexer = lexer.MustStateful(lexer.Rules{
"Root": {
{Name: "Escaped", Pattern: `\\[\{\}\[\]"\\]`, Action: nil},
{Name: "Markup", Pattern: `\[`, Action: lexer.Push("Markup")},
{Name: "Subst", Pattern: `{`, Action: lexer.Push("Subst")},
{Name: "Char", Pattern: `[%\{\["\\]|[^%\{\["\\]+`, Action: nil},
},
"Markup": {
{Name: "Whitespace", Pattern: `\s+`, Action: nil},
{Name: "Slash", Pattern: `/`, Action: nil},
{Name: "Ident", Pattern: `\w+`, Action: nil},
{Name: "Equals", Pattern: `=`, Action: nil},
{Name: "Subst", Pattern: `{`, Action: lexer.Push("Subst")},
{Name: "String", Pattern: `"`, Action: lexer.Push("String")},
{Name: "MarkupEnd", Pattern: `\]`, Action: lexer.Pop()},
},
"Subst": {
{Name: "Index", Pattern: `\d+`, Action: nil},
{Name: "SubstEnd", Pattern: `}`, Action: lexer.Pop()},
},
"String": {
{Name: "StringEnd", Pattern: `"`, Action: lexer.Pop()},
lexer.Include("Root"),
},
})
// a line is a kind of string, just missing the quotes...
lineParser = participle.MustBuild[parsedString](
participle.Lexer(lineLexer),
participle.Elide("Whitespace"),
)
)
// parsedString is used for both entire lines and the contents of double-quoted
// strings.
type parsedString struct {
Fragments []*fragment `parser:"@@*"`
}
// fragment is part of a string or line. The parser breaks it into pieces so
// that special pieces (escape sequences, markup, substitutions, and %) can be
// processed in a special way.
type fragment struct {
Escaped string `parser:"@Escaped"`
Markup *parsedMarkupTag `parser:"| Markup @@ MarkupEnd"`
Subst string `parser:"| Subst @Index SubstEnd"`
Text string `parser:"| @Char"`
}
// stringOrSubst appears inside markup tags. The value={0} prop is emitted
// without quoting the substitution token. Other props are usually of the form
// key="value".
type stringOrSubst struct {
String *parsedString `parser:"String @@ StringEnd"`
Subst string `parser:" | Subst @Index SubstEnd"`
}
// parsedMarkupTag is used for both format functions (select, plural, ordinal) and
// BBCode-esque markup tags ([b]Bold!?[/b]).
type parsedMarkupTag struct {
OpeningSlash string `parser:"@Slash?"` // indicates closing tag of a pair
Name string `parser:"@Ident?"` // used for all except close-all tag [/]
Props []*parsedProp `parser:"@@*"` // optional key="value" or value={0} properties
ClosingSlash string `parser:"@Slash?"` // indicates self-closing tag
}
// parsedProp is used for key="value" properties of format funcs and markup
// tags.
type parsedProp struct {
Key string `parser:"@Ident Equals"`
Value *stringOrSubst `parser:"@@"` // for ordinary values
}
type lineRenderer struct {
builder strings.Builder
attribs map[int][]*Attribute // lazily created; position -> tag event
open map[string][]*Attribute // lazily created; name -> stack of tags currently open
substs []string
lang language.Tag
}
func (b *lineRenderer) attStr() *AttributedString {
return &AttributedString{
str: b.builder.String(),
atts: b.attribs,
}
}
func (b *lineRenderer) openTag(name string, props []*parsedProp) error {
// Render each prop value into its own string, and put into a map
var m map[string]string
if len(props) > 0 {
m = make(map[string]string)
for _, prop := range props {
v, err := b.evalStringOrSubst(prop.Value)
if err != nil {
return err
}
m[prop.Key] = v
}
}
a := &Attribute{
Start: b.builder.Len(),
Name: name,
Props: m,
}
if b.open == nil {
b.open = make(map[string][]*Attribute)
}
if b.attribs == nil {
b.attribs = make(map[int][]*Attribute)
}
b.open[name] = append(b.open[name], a)
b.attribs[a.Start] = append(b.attribs[a.Start], a)
return nil
}
func (b *lineRenderer) closeTag(name string) error {
if b.open == nil {
return fmt.Errorf("tag %q not open", name)
}
as := b.open[name]
l := len(as)
if l == 0 {
return fmt.Errorf("tag %q not open", name)
}
// Close the most recent one
a, as := as[l-1], as[:l-1]
b.open[name] = as
a.End = b.builder.Len()
if a.Start == a.End {
// a is already in b.attribs[a.End]
return nil
}
b.attribs[a.End] = append(b.attribs[a.End], a)
return nil
}
func (b *lineRenderer) closeAll() {
for name, as := range b.open {
for _, a := range as {
a.End = b.builder.Len()
b.attribs[a.End] = append(b.attribs[a.End], a)
}
delete(b.open, name)
}
}
func (b *lineRenderer) renderString(p *parsedString) error {
for _, f := range p.Fragments {
if err := b.renderFragment(f); err != nil {
return err
}
}
return nil
}
func (b *lineRenderer) renderFragment(s *fragment) error {
if s == nil {
return nil
}
switch {
case s.Escaped != "":
b.builder.WriteString(s.Escaped[1:])
case s.Markup != nil:
return b.renderMarkupTag(s.Markup)
case s.Subst != "":
b.builder.WriteString(b.evalSubst(s.Subst))
default:
b.builder.WriteString(s.Text)
}
return nil
}
func (b *lineRenderer) evalSubst(index string) string {
n, err := strconv.Atoi(index)
if err != nil || n < 0 || n >= len(b.substs) {
return "{" + index + "}"
}
return b.substs[n]
}
// maps plural.Form values to identifiers used in Yarn Spinner plural and
// ordinal format functions
var formKeyTable = []string{
plural.Other: "other",
plural.Zero: "zero",
plural.One: "one",
plural.Two: "two",
plural.Few: "few",
plural.Many: "many",
}
func (b *lineRenderer) renderMarkupTag(f *parsedMarkupTag) error {
switch {
case f.Name == "select":
// [select value={0} m="bro" f="sis" nb="doc" /]
return b.renderSelectFormatFunc(f)
case f.Name == "plural":
// [plural value={0} one="an apple" other="% apples" /]
return b.renderPluralFormatFunc(f, plural.Cardinal)
case f.Name == "ordinal":
// [ordinal value={0} one="%st" two="%nd" ... /]
return b.renderPluralFormatFunc(f, plural.Ordinal)
case f.OpeningSlash == "/" && f.Name == "":
// Close-all tag [/]
b.closeAll()
return nil
case f.OpeningSlash == "/":
// Close tag [/foo]
return b.closeTag(f.Name)
case f.ClosingSlash == "/":
// Self-closing tag [foo/]
if err := b.openTag(f.Name, f.Props); err != nil {
return err
}
return b.closeTag(f.Name)
case f.Name != "":
// Open tag [foo]
return b.openTag(f.Name, f.Props)
default:
// Uhhhhhh... [] ?
b.builder.WriteString("[]")
return nil
}
}
// evalValueValue returns the string value of the markup tag property called
// "value". This is used by format functions.
func (b *lineRenderer) evalValueValue(f *parsedMarkupTag) (string, error) {
// Find the value property.
val, err := b.propValueForKey(f, "value")
if err != nil {
return "", err
}
// Evaluate its value!
return b.evalStringOrSubst(val)
}
func (b *lineRenderer) renderSelectFormatFunc(f *parsedMarkupTag) error {
// Get the value of the "value" property.
input, err := b.evalValueValue(f)
if err != nil {
return err
}
// Use that value to find the matching property.
val, err := b.propValueForKey(f, input)
if err != nil {
return err
}
// Render that value to the output!
return b.renderFormatFuncValue(val, input)
}
func (b *lineRenderer) renderPluralFormatFunc(f *parsedMarkupTag, rules *plural.Rules) error {
// Get the value of the "value" property.
input, err := b.evalValueValue(f)
if err != nil {
return err
}
// Use that value to match the cardinal form.
ops, err := cldr.NewOperands(input)
if err != nil {
return err
}
form := rules.MatchPlural(b.lang, int(ops.I), int(ops.V), int(ops.W), int(ops.F), int(ops.T))
if int(form) > len(formKeyTable) {
return fmt.Errorf("plural form %v not supported", form)
}
// Find the plural form in the properties.
val, err := b.propValueForKey(f, formKeyTable[form])
if err != nil {
return err
}
// Render that value to the output!
return b.renderFormatFuncValue(val, input)
}
func (b *lineRenderer) evalStringOrSubst(s *stringOrSubst) (string, error) {
if s.Subst != "" {
return b.evalSubst(s.Subst), nil
}
inb := &lineRenderer{
substs: b.substs,
lang: b.lang,
}
if err := inb.renderString(s.String); err != nil {
return "", err
}
return inb.builder.String(), nil
}
// propValueForKey searches f.Props for the option matching the key, and
// then returns the Value.
func (b *lineRenderer) propValueForKey(f *parsedMarkupTag, key string) (*stringOrSubst, error) {
for _, opt := range f.Props {
if opt.Key == key {
return opt.Value, nil
}
}
return nil, fmt.Errorf("key %q not found in %#v", key, f.Props)
}
func (b *lineRenderer) renderFormatFuncValue(s *stringOrSubst, input string) error {
// Format func values have an additional token that needs to be processed
// specially (%).
if s.Subst != "" {
b.builder.WriteString(b.evalSubst(s.Subst))
return nil
}
for _, v := range s.String.Fragments {
if v.Text == "%" {
b.builder.WriteString(input)
continue
}
if err := b.renderFragment(v); err != nil {
return err
}
}
return nil
}