123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285 |
- package yu_url
- import (
- yu_const "gogs.qqck.cn/s/tools/const"
- )
- type encoding int
- const (
- encodePath encoding = 1 + iota
- encodePathSegment
- encodeHost
- encodeZone
- encodeUserPassword
- encodeQueryComponent
- encodeFragment
- )
- // QueryEscape escapes the string so it can be safely placed
- // inside a URL query.
- func QueryEscape(s string) string {
- return escape(s, encodeQueryComponent)
- }
- // PathEscape escapes the string so it can be safely placed inside a URL path segment,
- // replacing special characters (including /) with %XX sequences as needed.
- func PathEscape(s string) string {
- return escape(s, encodePathSegment)
- }
- func escape(s string, mode encoding) string {
- spaceCount, hexCount := 0, 0
- for i := 0; i < len(s); i++ {
- c := s[i]
- if shouldEscape(c, mode) {
- if c == ' ' && mode == encodeQueryComponent {
- spaceCount++
- } else {
- hexCount++
- }
- }
- }
- if spaceCount == 0 && hexCount == 0 {
- return s
- }
- var buf [64]byte
- var t []byte
- required := len(s) + 2*hexCount
- if required <= len(buf) {
- t = buf[:required]
- } else {
- t = make([]byte, required)
- }
- if hexCount == 0 {
- copy(t, s)
- for i := 0; i < len(s); i++ {
- if s[i] == ' ' {
- t[i] = '+'
- }
- }
- return string(t)
- }
- j := 0
- for i := 0; i < len(s); i++ {
- switch c := s[i]; {
- case c == ' ' && mode == encodeQueryComponent:
- t[j] = '+'
- j++
- case shouldEscape(c, mode):
- t[j] = '%'
- t[j+1] = yu_const.HexUpper[c>>4]
- t[j+2] = yu_const.HexUpper[c&15]
- j += 3
- default:
- t[j] = s[i]
- j++
- }
- }
- return string(t)
- }
- // Return true if the specified character should be escaped when
- // appearing in a URL string, according to RFC 3986.
- //
- // Please be informed that for now shouldEscape does not check all
- // reserved characters correctly. See golang.org/issue/5684.
- func shouldEscape(c byte, mode encoding) bool {
- // §2.3 Unreserved characters (alphanum)
- if 'a' <= c && c <= 'z' || 'A' <= c && c <= 'Z' || '0' <= c && c <= '9' {
- return false
- }
- if mode == encodeHost || mode == encodeZone {
- // §3.2.2 Host allows
- // sub-delims = "!" / "$" / "&" / "'" / "(" / ")" / "*" / "+" / "," / ";" / "="
- // as part of reg-name.
- // We add : because we include :port as part of host.
- // We add [ ] because we include [ipv6]:port as part of host.
- // We add < > because they're the only characters left that
- // we could possibly allow, and Parse will reject them if we
- // escape them (because hosts can't use %-encoding for
- // ASCII bytes).
- switch c {
- case '!', '$', '&', '\'', '(', ')', '*', '+', ',', ';', '=', ':', '[', ']', '<', '>', '"':
- return false
- }
- }
- switch c {
- case '-', '_', '.', '~': // §2.3 Unreserved characters (mark)
- return false
- case '$', '&', '+', ',', '/', ':', ';', '=', '?', '@': // §2.2 Reserved characters (reserved)
- // Different sections of the URL allow a few of
- // the reserved characters to appear unescaped.
- switch mode {
- case encodePath: // §3.3
- // The RFC allows : @ & = + $ but saves / ; , for assigning
- // meaning to individual path segments. This package
- // only manipulates the path as a whole, so we allow those
- // last three as well. That leaves only ? to escape.
- return c == '?'
- case encodePathSegment: // §3.3
- // The RFC allows : @ & = + $ but saves / ; , for assigning
- // meaning to individual path segments.
- return c == '/' || c == ';' || c == ',' || c == '?'
- case encodeUserPassword: // §3.2.1
- // The RFC allows ';', ':', '&', '=', '+', '$', and ',' in
- // userinfo, so we must escape only '@', '/', and '?'.
- // The parsing of userinfo treats ':' as special so we must escape
- // that too.
- return c == '@' || c == '/' || c == '?' || c == ':'
- case encodeQueryComponent: // §3.4
- // The RFC reserves (so we must escape) everything.
- return true
- case encodeFragment: // §4.1
- // The RFC text is silent but the grammar allows
- // everything, so escape nothing.
- return false
- }
- }
- if mode == encodeFragment {
- // RFC 3986 §2.2 allows not escaping sub-delims. A subset of sub-delims are
- // included in reserved from RFC 2396 §2.2. The remaining sub-delims do not
- // need to be escaped. To minimize potential breakage, we apply two restrictions:
- // (1) we always escape sub-delims outside of the fragment, and (2) we always
- // escape single quote to avoid breaking callers that had previously assumed that
- // single quotes would be escaped. See issue #19917.
- switch c {
- case '!', '(', ')', '*':
- return false
- }
- }
- // Everything else must be escaped.
- return true
- }
- // QueryUnescape does the inverse transformation of QueryEscape,
- // converting each 3-byte encoded substring of the form "%AB" into the
- // hex-decoded byte 0xAB.
- // It returns an error if any % is not followed by two hexadecimal
- // digits.
- func QueryUnescape(s string) string {
- return unescape(s, encodeQueryComponent)
- }
- // PathUnescape does the inverse transformation of PathEscape,
- // converting each 3-byte encoded substring of the form "%AB" into the
- // hex-decoded byte 0xAB. It returns an error if any % is not followed
- // by two hexadecimal digits.
- //
- // PathUnescape is identical to QueryUnescape except that it does not
- // unescape '+' to ' ' (space).
- func PathUnescape(s string) string {
- return unescape(s, encodePathSegment)
- }
- // unescape unescapes a string; the mode specifies
- // which section of the URL string is being unescaped.
- func unescape(s string, mode encoding) string {
- // Count %, check that they're well-formed.
- n := 0
- hasPlus := false
- for i := 0; i < len(s); {
- switch s[i] {
- case '%':
- n++
- if i+2 >= len(s) || !ishex(s[i+1]) || !ishex(s[i+2]) {
- s = s[i:]
- if len(s) > 3 {
- s = s[:3]
- }
- return ""
- }
- // Per https://tools.ietf.org/html/rfc3986#page-21
- // in the host component %-encoding can only be used
- // for non-ASCII bytes.
- // But https://tools.ietf.org/html/rfc6874#section-2
- // introduces %25 being allowed to escape a percent sign
- // in IPv6 scoped-address literals. Yay.
- if mode == encodeHost && unhex(s[i+1]) < 8 && s[i:i+3] != "%25" {
- return ""
- }
- if mode == encodeZone {
- // RFC 6874 says basically "anything goes" for zone identifiers
- // and that even non-ASCII can be redundantly escaped,
- // but it seems prudent to restrict %-escaped bytes here to those
- // that are valid host name bytes in their unescaped form.
- // That is, you can use escaping in the zone identifier but not
- // to introduce bytes you couldn't just write directly.
- // But Windows puts spaces here! Yay.
- v := unhex(s[i+1])<<4 | unhex(s[i+2])
- if s[i:i+3] != "%25" && v != ' ' && shouldEscape(v, encodeHost) {
- return ""
- }
- }
- i += 3
- case '+':
- hasPlus = mode == encodeQueryComponent
- i++
- default:
- if (mode == encodeHost || mode == encodeZone) && s[i] < 0x80 && shouldEscape(s[i], mode) {
- return ""
- }
- i++
- }
- }
- if n == 0 && !hasPlus {
- return s
- }
- t := make([]byte, 0, len(s)-2*n)
- for i := 0; i < len(s); i++ {
- switch s[i] {
- case '%':
- t = append(t, unhex(s[i+1])<<4|unhex(s[i+2]))
- i += 2
- case '+':
- if mode == encodeQueryComponent {
- t = append(t, ' ')
- } else {
- t = append(t, '+')
- }
- default:
- t = append(t, s[i])
- }
- }
- return string(t)
- }
- func ishex(c byte) bool {
- switch {
- case '0' <= c && c <= '9':
- return true
- case 'a' <= c && c <= 'f':
- return true
- case 'A' <= c && c <= 'F':
- return true
- }
- return false
- }
- func unhex(c byte) byte {
- switch {
- case '0' <= c && c <= '9':
- return c - '0'
- case 'a' <= c && c <= 'f':
- return c - 'a' + 10
- case 'A' <= c && c <= 'F':
- return c - 'A' + 10
- }
- return 0
- }
|