You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
157 lines
4.1 KiB
157 lines
4.1 KiB
// Copyright 2015 The Go Authors. All rights reserved. |
|
// Use of this source code is governed by a BSD-style |
|
// license that can be found in the LICENSE file. |
|
|
|
package precis |
|
|
|
import ( |
|
"golang.org/x/text/cases" |
|
"golang.org/x/text/language" |
|
"golang.org/x/text/runes" |
|
"golang.org/x/text/transform" |
|
"golang.org/x/text/unicode/norm" |
|
) |
|
|
|
// An Option is used to define the behavior and rules of a Profile. |
|
type Option func(*options) |
|
|
|
type options struct { |
|
// Preparation options |
|
foldWidth bool |
|
|
|
// Enforcement options |
|
asciiLower bool |
|
cases transform.SpanningTransformer |
|
disallow runes.Set |
|
norm transform.SpanningTransformer |
|
additional []func() transform.SpanningTransformer |
|
width transform.SpanningTransformer |
|
disallowEmpty bool |
|
bidiRule bool |
|
repeat bool |
|
|
|
// Comparison options |
|
ignorecase bool |
|
} |
|
|
|
func getOpts(o ...Option) (res options) { |
|
for _, f := range o { |
|
f(&res) |
|
} |
|
// Using a SpanningTransformer, instead of norm.Form prevents an allocation |
|
// down the road. |
|
if res.norm == nil { |
|
res.norm = norm.NFC |
|
} |
|
return |
|
} |
|
|
|
var ( |
|
// The IgnoreCase option causes the profile to perform a case insensitive |
|
// comparison during the PRECIS comparison step. |
|
IgnoreCase Option = ignoreCase |
|
|
|
// The FoldWidth option causes the profile to map non-canonical wide and |
|
// narrow variants to their decomposition mapping. This is useful for |
|
// profiles that are based on the identifier class which would otherwise |
|
// disallow such characters. |
|
FoldWidth Option = foldWidth |
|
|
|
// The DisallowEmpty option causes the enforcement step to return an error if |
|
// the resulting string would be empty. |
|
DisallowEmpty Option = disallowEmpty |
|
|
|
// The BidiRule option causes the Bidi Rule defined in RFC 5893 to be |
|
// applied. |
|
BidiRule Option = bidiRule |
|
) |
|
|
|
var ( |
|
ignoreCase = func(o *options) { |
|
o.ignorecase = true |
|
} |
|
foldWidth = func(o *options) { |
|
o.foldWidth = true |
|
} |
|
disallowEmpty = func(o *options) { |
|
o.disallowEmpty = true |
|
} |
|
bidiRule = func(o *options) { |
|
o.bidiRule = true |
|
} |
|
repeat = func(o *options) { |
|
o.repeat = true |
|
} |
|
) |
|
|
|
// TODO: move this logic to package transform |
|
|
|
type spanWrap struct{ transform.Transformer } |
|
|
|
func (s spanWrap) Span(src []byte, atEOF bool) (n int, err error) { |
|
return 0, transform.ErrEndOfSpan |
|
} |
|
|
|
// TODO: allow different types? For instance: |
|
// func() transform.Transformer |
|
// func() transform.SpanningTransformer |
|
// func([]byte) bool // validation only |
|
// |
|
// Also, would be great if we could detect if a transformer is reentrant. |
|
|
|
// The AdditionalMapping option defines the additional mapping rule for the |
|
// Profile by applying Transformer's in sequence. |
|
func AdditionalMapping(t ...func() transform.Transformer) Option { |
|
return func(o *options) { |
|
for _, f := range t { |
|
sf := func() transform.SpanningTransformer { |
|
return f().(transform.SpanningTransformer) |
|
} |
|
if _, ok := f().(transform.SpanningTransformer); !ok { |
|
sf = func() transform.SpanningTransformer { |
|
return spanWrap{f()} |
|
} |
|
} |
|
o.additional = append(o.additional, sf) |
|
} |
|
} |
|
} |
|
|
|
// The Norm option defines a Profile's normalization rule. Defaults to NFC. |
|
func Norm(f norm.Form) Option { |
|
return func(o *options) { |
|
o.norm = f |
|
} |
|
} |
|
|
|
// The FoldCase option defines a Profile's case mapping rule. Options can be |
|
// provided to determine the type of case folding used. |
|
func FoldCase(opts ...cases.Option) Option { |
|
return func(o *options) { |
|
o.asciiLower = true |
|
o.cases = cases.Fold(opts...) |
|
} |
|
} |
|
|
|
// The LowerCase option defines a Profile's case mapping rule. Options can be |
|
// provided to determine the type of case folding used. |
|
func LowerCase(opts ...cases.Option) Option { |
|
return func(o *options) { |
|
o.asciiLower = true |
|
if len(opts) == 0 { |
|
o.cases = cases.Lower(language.Und, cases.HandleFinalSigma(false)) |
|
return |
|
} |
|
|
|
opts = append([]cases.Option{cases.HandleFinalSigma(false)}, opts...) |
|
o.cases = cases.Lower(language.Und, opts...) |
|
} |
|
} |
|
|
|
// The Disallow option further restricts a Profile's allowed characters beyond |
|
// what is disallowed by the underlying string class. |
|
func Disallow(set runes.Set) Option { |
|
return func(o *options) { |
|
o.disallow = set |
|
} |
|
}
|
|
|