mirror of
https://github.com/gohugoio/hugo.git
synced 2024-11-07 20:30:36 -05:00
c2e2913872
And looks even faster: Compared to previous attempt: benchmark old ns/op new ns/op delta BenchmarkAbsUrl 30902 27206 -11.96% BenchmarkXmlAbsUrl 15389 14216 -7.62% benchmark old allocs new allocs delta BenchmarkAbsUrl 33 28 -15.15% BenchmarkXmlAbsUrl 16 14 -12.50% benchmark old bytes new bytes delta BenchmarkAbsUrl 4167 3504 -15.91% BenchmarkXmlAbsUrl 2057 2048 -0.44% Compared to before I started all of this: benchmark old ns/op new ns/op delta BenchmarkAbsUrl 36219 27206 -24.88% BenchmarkXmlAbsUrl 23903 14216 -40.53% benchmark old allocs new allocs delta BenchmarkAbsUrl 60 28 -53.33% BenchmarkXmlAbsUrl 30 14 -53.33% benchmark old bytes new bytes delta BenchmarkAbsUrl 5842 3504 -40.02% BenchmarkXmlAbsUrl 3754 2048 -45.44% Fixes #816
289 lines
5.5 KiB
Go
289 lines
5.5 KiB
Go
package transform
|
|
|
|
import (
|
|
"bytes"
|
|
bp "github.com/spf13/hugo/bufferpool"
|
|
"net/url"
|
|
"strings"
|
|
"sync"
|
|
"unicode/utf8"
|
|
)
|
|
|
|
// position (in bytes)
|
|
type pos int
|
|
|
|
type matchState int
|
|
|
|
const (
|
|
matchStateNone matchState = iota
|
|
matchStateWhitespace
|
|
matchStatePartial
|
|
matchStateFull
|
|
)
|
|
|
|
type item struct {
|
|
typ itemType
|
|
pos pos
|
|
val []byte
|
|
}
|
|
|
|
type itemType int
|
|
|
|
const (
|
|
tText itemType = iota
|
|
|
|
// matches
|
|
tSrcdq
|
|
tHrefdq
|
|
tSrcsq
|
|
tHrefsq
|
|
)
|
|
|
|
type contentlexer struct {
|
|
content []byte
|
|
|
|
pos pos // input position
|
|
start pos // item start position
|
|
width pos // width of last element
|
|
|
|
matchers []absurlMatcher
|
|
state stateFunc
|
|
prefixLookup *prefixes
|
|
|
|
// items delivered to client
|
|
items []item
|
|
}
|
|
|
|
type stateFunc func(*contentlexer) stateFunc
|
|
|
|
type prefixRunes []rune
|
|
|
|
type prefixes struct {
|
|
pr []prefixRunes
|
|
curr prefixRunes // current prefix lookup table
|
|
i int // current index
|
|
|
|
// first rune in potential match
|
|
first rune
|
|
|
|
// match-state:
|
|
// none, whitespace, partial, full
|
|
ms matchState
|
|
}
|
|
|
|
// match returns partial and full match for the prefix in play
|
|
// - it's a full match if all prefix runes has checked out in row
|
|
// - it's a partial match if it's on its way towards a full match
|
|
func (l *contentlexer) match(r rune) {
|
|
p := l.prefixLookup
|
|
if p.curr == nil {
|
|
// assumes prefixes all start off on a different rune
|
|
// works in this special case: href, src
|
|
p.i = 0
|
|
for _, pr := range p.pr {
|
|
if pr[p.i] == r {
|
|
fullMatch := len(p.pr) == 1
|
|
p.first = r
|
|
if !fullMatch {
|
|
p.curr = pr
|
|
l.prefixLookup.ms = matchStatePartial
|
|
} else {
|
|
l.prefixLookup.ms = matchStateFull
|
|
}
|
|
return
|
|
}
|
|
}
|
|
} else {
|
|
p.i++
|
|
if p.curr[p.i] == r {
|
|
fullMatch := len(p.curr) == p.i+1
|
|
if fullMatch {
|
|
p.curr = nil
|
|
l.prefixLookup.ms = matchStateFull
|
|
} else {
|
|
l.prefixLookup.ms = matchStatePartial
|
|
}
|
|
return
|
|
}
|
|
|
|
p.curr = nil
|
|
}
|
|
|
|
l.prefixLookup.ms = matchStateNone
|
|
}
|
|
|
|
func (l *contentlexer) emit(t itemType) {
|
|
l.items = append(l.items, item{t, l.start, l.content[l.start:l.pos]})
|
|
l.start = l.pos
|
|
}
|
|
|
|
var mainPrefixRunes = []prefixRunes{{'s', 'r', 'c', '='}, {'h', 'r', 'e', 'f', '='}}
|
|
|
|
var itemSlicePool = &sync.Pool{
|
|
New: func() interface{} {
|
|
return make([]item, 0, 8)
|
|
},
|
|
}
|
|
|
|
func (l *contentlexer) runReplacer() {
|
|
for l.state = lexReplacements; l.state != nil; {
|
|
l.state = l.state(l)
|
|
}
|
|
}
|
|
|
|
type absurlMatcher struct {
|
|
replaceType itemType
|
|
match []byte
|
|
replacement []byte
|
|
}
|
|
|
|
func (a absurlMatcher) isSourceType() bool {
|
|
return a.replaceType == tSrcdq || a.replaceType == tSrcsq
|
|
}
|
|
|
|
func lexReplacements(l *contentlexer) stateFunc {
|
|
contentLength := len(l.content)
|
|
var r rune
|
|
|
|
for {
|
|
if int(l.pos) >= contentLength {
|
|
l.width = 0
|
|
break
|
|
}
|
|
|
|
var width int = 1
|
|
r = rune(l.content[l.pos])
|
|
if r >= utf8.RuneSelf {
|
|
r, width = utf8.DecodeRune(l.content[l.pos:])
|
|
}
|
|
l.width = pos(width)
|
|
l.pos += l.width
|
|
|
|
if r == ' ' {
|
|
l.prefixLookup.ms = matchStateWhitespace
|
|
} else if l.prefixLookup.ms != matchStateNone {
|
|
l.match(r)
|
|
if l.prefixLookup.ms == matchStateFull {
|
|
checkCandidate(l)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// Done!
|
|
if l.pos > l.start {
|
|
l.emit(tText)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func checkCandidate(l *contentlexer) {
|
|
isSource := l.prefixLookup.first == 's'
|
|
for _, m := range l.matchers {
|
|
|
|
if isSource && !m.isSourceType() || !isSource && m.isSourceType() {
|
|
continue
|
|
}
|
|
|
|
if bytes.HasPrefix(l.content[l.pos:], m.match) {
|
|
// check for schemaless urls
|
|
posAfter := pos(int(l.pos) + len(m.match))
|
|
if int(posAfter) >= len(l.content) {
|
|
return
|
|
}
|
|
r, _ := utf8.DecodeRune(l.content[posAfter:])
|
|
if r == '/' {
|
|
// schemaless: skip
|
|
return
|
|
}
|
|
if l.pos > l.start {
|
|
l.emit(tText)
|
|
}
|
|
l.pos += pos(len(m.match))
|
|
l.emit(m.replaceType)
|
|
return
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
func doReplace(content []byte, matchers []absurlMatcher) []byte {
|
|
b := bp.GetBuffer()
|
|
defer bp.PutBuffer(b)
|
|
|
|
var items []item
|
|
if x := itemSlicePool.Get(); x != nil {
|
|
items = x.([]item)[:0]
|
|
defer itemSlicePool.Put(items)
|
|
} else {
|
|
items = make([]item, 0, 8)
|
|
}
|
|
|
|
lexer := &contentlexer{content: content,
|
|
items: items,
|
|
prefixLookup: &prefixes{pr: mainPrefixRunes},
|
|
matchers: matchers}
|
|
|
|
lexer.runReplacer()
|
|
|
|
for _, token := range lexer.items {
|
|
switch token.typ {
|
|
case tText:
|
|
b.Write(token.val)
|
|
default:
|
|
for _, e := range matchers {
|
|
if token.typ == e.replaceType {
|
|
b.Write(e.replacement)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return b.Bytes()
|
|
}
|
|
|
|
type absurlReplacer struct {
|
|
htmlMatchers []absurlMatcher
|
|
xmlMatchers []absurlMatcher
|
|
}
|
|
|
|
func newAbsurlReplacer(baseUrl string) *absurlReplacer {
|
|
u, _ := url.Parse(baseUrl)
|
|
base := strings.TrimRight(u.String(), "/")
|
|
|
|
// HTML
|
|
dqHtmlMatch := []byte("\"/")
|
|
sqHtmlMatch := []byte("'/")
|
|
|
|
// XML
|
|
dqXmlMatch := []byte(""/")
|
|
sqXmlMatch := []byte("'/")
|
|
|
|
dqHtml := []byte("\"" + base + "/")
|
|
sqHtml := []byte("'" + base + "/")
|
|
|
|
dqXml := []byte(""" + base + "/")
|
|
sqXml := []byte("'" + base + "/")
|
|
|
|
return &absurlReplacer{htmlMatchers: []absurlMatcher{
|
|
{tSrcdq, dqHtmlMatch, dqHtml},
|
|
{tSrcsq, sqHtmlMatch, sqHtml},
|
|
{tHrefdq, dqHtmlMatch, dqHtml},
|
|
{tHrefsq, sqHtmlMatch, sqHtml}},
|
|
xmlMatchers: []absurlMatcher{
|
|
{tSrcdq, dqXmlMatch, dqXml},
|
|
{tSrcsq, sqXmlMatch, sqXml},
|
|
{tHrefdq, dqXmlMatch, dqXml},
|
|
{tHrefsq, sqXmlMatch, sqXml},
|
|
}}
|
|
|
|
}
|
|
|
|
func (au *absurlReplacer) replaceInHtml(content []byte) []byte {
|
|
return doReplace(content, au.htmlMatchers)
|
|
}
|
|
|
|
func (au *absurlReplacer) replaceInXml(content []byte) []byte {
|
|
return doReplace(content, au.xmlMatchers)
|
|
}
|