2016-12-07 21:16:30 +00:00
|
|
|
// +build !cvis
|
2016-08-25 18:17:08 +00:00
|
|
|
|
|
|
|
package mtree
|
|
|
|
|
|
|
|
import "unicode"
|
|
|
|
|
|
|
|
func unvis(src string) (string, error) {
|
2017-02-10 12:08:00 +00:00
|
|
|
dst := []rune{}
|
2016-08-25 18:17:08 +00:00
|
|
|
var s state
|
|
|
|
for i, r := range src {
|
|
|
|
again:
|
2017-02-10 12:08:00 +00:00
|
|
|
err := unvisRune(&dst, r, &s, 0)
|
2016-08-25 18:17:08 +00:00
|
|
|
switch err {
|
|
|
|
case unvisValid:
|
|
|
|
break
|
|
|
|
case unvisValidPush:
|
|
|
|
goto again
|
|
|
|
case unvisNone:
|
|
|
|
fallthrough
|
|
|
|
case unvisNochar:
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if i == len(src)-1 {
|
2017-02-10 12:08:00 +00:00
|
|
|
unvisRune(&dst, r, &s, unvisEnd)
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
}
|
2017-02-10 12:08:00 +00:00
|
|
|
|
|
|
|
str := ""
|
|
|
|
for _, ch := range dst {
|
|
|
|
str += string(ch)
|
|
|
|
}
|
|
|
|
return str, nil
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
|
2017-02-10 12:08:00 +00:00
|
|
|
func unvisRune(dst *[]rune, r rune, s *state, flags VisFlag) error {
|
2016-08-25 18:17:08 +00:00
|
|
|
if (flags & unvisEnd) != 0 {
|
|
|
|
if *s == stateOctal2 || *s == stateOctal3 {
|
|
|
|
*s = stateGround
|
|
|
|
return unvisValid
|
|
|
|
}
|
|
|
|
if *s == stateGround {
|
|
|
|
return unvisNochar
|
|
|
|
}
|
|
|
|
return unvisErrSynbad
|
|
|
|
}
|
|
|
|
|
|
|
|
switch *s & ^stateHTTP {
|
|
|
|
case stateGround:
|
|
|
|
if r == '\\' {
|
|
|
|
*s = stateStart
|
|
|
|
return unvisNone
|
|
|
|
}
|
|
|
|
if flags&VisHttpstyle != 0 && r == '%' {
|
|
|
|
*s = stateStart | stateHTTP
|
|
|
|
return unvisNone
|
|
|
|
}
|
2017-02-10 12:08:00 +00:00
|
|
|
*dst = append(*dst, r)
|
2016-08-25 18:17:08 +00:00
|
|
|
return unvisValid
|
|
|
|
case stateStart:
|
|
|
|
if *s&stateHTTP != 0 && ishex(unicode.ToLower(r)) {
|
|
|
|
if unicode.IsNumber(r) {
|
2017-02-10 12:08:00 +00:00
|
|
|
*dst = append(*dst, r-'0')
|
2016-08-25 18:17:08 +00:00
|
|
|
} else {
|
2017-02-10 12:08:00 +00:00
|
|
|
*dst = append(*dst, unicode.ToLower(r)-'a')
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
*s = stateHex2
|
|
|
|
return unvisNone
|
|
|
|
}
|
|
|
|
switch r {
|
|
|
|
case '\\':
|
|
|
|
*s = stateGround
|
2017-02-10 12:08:00 +00:00
|
|
|
*dst = append(*dst, r)
|
2016-08-25 18:17:08 +00:00
|
|
|
return unvisValid
|
|
|
|
case '0':
|
|
|
|
fallthrough
|
|
|
|
case '1':
|
|
|
|
fallthrough
|
|
|
|
case '2':
|
|
|
|
fallthrough
|
|
|
|
case '3':
|
|
|
|
fallthrough
|
|
|
|
case '4':
|
|
|
|
fallthrough
|
|
|
|
case '5':
|
|
|
|
fallthrough
|
|
|
|
case '6':
|
|
|
|
fallthrough
|
|
|
|
case '7':
|
|
|
|
*s = stateOctal2
|
2017-02-10 12:08:00 +00:00
|
|
|
*dst = append(*dst, r-'0')
|
2016-08-25 18:17:08 +00:00
|
|
|
return unvisNone
|
|
|
|
case 'M':
|
|
|
|
*s = stateMeta
|
2017-02-10 12:08:00 +00:00
|
|
|
*dst = append(*dst, rune(0200))
|
2016-08-25 18:17:08 +00:00
|
|
|
return unvisNone
|
|
|
|
case '^':
|
|
|
|
*s = stateCtrl
|
|
|
|
return unvisNone
|
|
|
|
case 'n':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\n')
|
|
|
|
return unvisValid
|
|
|
|
case 'r':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\r')
|
|
|
|
return unvisValid
|
|
|
|
case 'b':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\b')
|
|
|
|
return unvisValid
|
|
|
|
case 'a':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\007')
|
|
|
|
return unvisValid
|
|
|
|
case 'v':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\v')
|
|
|
|
return unvisValid
|
|
|
|
case 't':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\t')
|
|
|
|
return unvisValid
|
|
|
|
case 'f':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\f')
|
|
|
|
return unvisValid
|
|
|
|
case 's':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, ' ')
|
|
|
|
return unvisValid
|
|
|
|
case 'E':
|
|
|
|
*s = stateGround
|
|
|
|
*dst = append(*dst, '\033')
|
|
|
|
return unvisValid
|
|
|
|
case '\n':
|
|
|
|
// hidden newline
|
|
|
|
*s = stateGround
|
|
|
|
return unvisNochar
|
|
|
|
case '$':
|
|
|
|
// hidden marker
|
|
|
|
*s = stateGround
|
|
|
|
return unvisNochar
|
|
|
|
}
|
|
|
|
*s = stateGround
|
|
|
|
return unvisErrSynbad
|
|
|
|
case stateMeta:
|
|
|
|
if r == '-' {
|
|
|
|
*s = stateMeta1
|
|
|
|
} else if r == '^' {
|
|
|
|
*s = stateCtrl
|
|
|
|
} else {
|
|
|
|
*s = stateGround
|
|
|
|
return unvisErrSynbad
|
|
|
|
}
|
|
|
|
return unvisNone
|
|
|
|
case stateMeta1:
|
|
|
|
*s = stateGround
|
|
|
|
dp := *dst
|
2017-02-10 12:08:00 +00:00
|
|
|
dp[len(dp)-1] |= r
|
2016-08-25 18:17:08 +00:00
|
|
|
return unvisValid
|
|
|
|
case stateCtrl:
|
|
|
|
dp := *dst
|
|
|
|
if r == '?' {
|
2017-02-10 12:08:00 +00:00
|
|
|
dp[len(dp)-1] |= rune(0177)
|
2016-08-25 18:17:08 +00:00
|
|
|
} else {
|
2017-02-10 12:08:00 +00:00
|
|
|
dp[len(dp)-1] |= r & 037
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
*s = stateGround
|
|
|
|
return unvisValid
|
|
|
|
case stateOctal2:
|
|
|
|
if isoctal(r) {
|
|
|
|
dp := *dst
|
|
|
|
if len(dp) > 0 {
|
|
|
|
last := dp[len(dp)-1]
|
2017-02-10 12:08:00 +00:00
|
|
|
dp[len(dp)-1] = (last << 3) + (r - '0')
|
2016-08-25 18:17:08 +00:00
|
|
|
} else {
|
2017-02-10 12:08:00 +00:00
|
|
|
dp = append(dp, (0<<3)+(r-'0'))
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
*s = stateOctal3
|
|
|
|
return unvisNone
|
|
|
|
}
|
|
|
|
*s = stateGround
|
|
|
|
return unvisValidPush
|
|
|
|
case stateOctal3:
|
|
|
|
*s = stateGround
|
|
|
|
if isoctal(r) {
|
|
|
|
dp := *dst
|
|
|
|
if len(dp) > 0 {
|
|
|
|
last := dp[len(dp)-1]
|
2017-02-10 12:08:00 +00:00
|
|
|
dp[len(dp)-1] = (last << 3) + (r - '0')
|
2016-08-25 18:17:08 +00:00
|
|
|
} else {
|
2017-02-10 12:08:00 +00:00
|
|
|
dp = append(dp, (0<<3)+(r-'0'))
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
return unvisValid
|
|
|
|
}
|
|
|
|
return unvisValidPush
|
|
|
|
case stateHex2:
|
|
|
|
if ishex(unicode.ToLower(r)) {
|
2017-02-10 12:08:00 +00:00
|
|
|
last := rune(0)
|
2016-08-25 18:17:08 +00:00
|
|
|
dp := *dst
|
|
|
|
if len(dp) > 0 {
|
|
|
|
last = dp[len(dp)-1]
|
|
|
|
}
|
|
|
|
if unicode.IsNumber(r) {
|
2017-02-10 12:08:00 +00:00
|
|
|
dp = append(dp, (last<<4)+(r-'0'))
|
2016-08-25 18:17:08 +00:00
|
|
|
} else {
|
2017-02-10 12:08:00 +00:00
|
|
|
dp = append(dp, (last<<4)+(unicode.ToLower(r)-'a'+10))
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
*s = stateGround
|
|
|
|
return unvisValid
|
|
|
|
}
|
|
|
|
|
2016-12-07 21:26:04 +00:00
|
|
|
*s = stateGround
|
|
|
|
return unvisErrSynbad
|
2016-08-25 18:17:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type state int
|
|
|
|
|
|
|
|
const (
|
|
|
|
stateGround state = iota /* haven't seen escape char */
|
|
|
|
stateStart /* start decoding special sequence */
|
|
|
|
stateMeta /* metachar started (M) */
|
|
|
|
stateMeta1 /* metachar more, regular char (-) */
|
|
|
|
stateCtrl /* control char started (^) */
|
|
|
|
stateOctal2 /* octal digit 2 */
|
|
|
|
stateOctal3 /* octal digit 3 */
|
|
|
|
stateHex2 /* hex digit 2 */
|
|
|
|
|
|
|
|
stateHTTP state = 0x080 /* %HEXHEX escape */
|
|
|
|
)
|