quoteString

pull/2136/head
merlin 2024-10-01 14:50:59 +03:00
parent 58d4c0c94f
commit ea1e13a660
No known key found for this signature in database
GPG Key ID: 7EDDCEA6A90062E0
2 changed files with 54 additions and 2 deletions

View File

@ -81,7 +81,7 @@ func (q *Query) Sanitize(args ...any) (string, error) {
case []byte:
p = quoteBytes(buf.AvailableBuffer(), arg)
case string:
p = []byte(QuoteString(arg))
p = quoteString(buf.AvailableBuffer(), arg)
case time.Time:
p = arg.Truncate(time.Microsecond).
AppendFormat(buf.AvailableBuffer(), "'2006-01-02 15:04:05.999999999Z07:00:00'")
@ -124,7 +124,34 @@ func NewQuery(sql string) (*Query, error) {
}
func QuoteString(str string) string {
return "'" + strings.ReplaceAll(str, "'", "''") + "'"
return string(quoteString(nil, str))
}
func quoteString(dst []byte, str string) []byte {
const quote = "'"
n := strings.Count(str, quote)
dst = append(dst, quote...)
p := slices.Grow(dst[len(dst):], len(str)+2*n)
for len(str) > 0 {
i := strings.Index(str, quote)
if i < 0 {
p = append(p, str...)
break
}
p = append(p, str[:i]...)
p = append(p, "''"...)
str = str[i+1:]
}
dst = append(dst, p...)
dst = append(dst, quote...)
return dst
}
func QuoteBytes(buf []byte) string {

View File

@ -2,6 +2,7 @@ package sanitize_test
import (
"encoding/hex"
"strings"
"testing"
"time"
@ -229,6 +230,30 @@ func TestQuerySanitize(t *testing.T) {
}
}
func TestQuoteString(t *testing.T) {
tc := func(name, input string) {
t.Run(name, func(t *testing.T) {
t.Parallel()
got := sanitize.QuoteString(input)
want := oldQuoteString(input)
if got != want {
t.Errorf("got: %s", got)
t.Fatalf("want: %s", want)
}
})
}
tc("empty", "")
tc("text", "abcd")
tc("with quotes", `one's hat is always a cat`)
}
func oldQuoteString(str string) string {
return "'" + strings.ReplaceAll(str, "'", "''") + "'"
}
func TestQuoteBytes(t *testing.T) {
tc := func(name string, input []byte) {
t.Run(name, func(t *testing.T) {