I believe I now have a working wikilink lexer
parent
4bc6d7ceda
commit
949b5c240e
@ -1,118 +1,130 @@
|
|||||||
package wikilink
|
package wikilink
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"log"
|
|
||||||
"strings"
|
"strings"
|
||||||
"unicode"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func lexFragment(l *Lexer) stateFn {
|
func isOpenLink(s string) bool {
|
||||||
log.Println("entering lexFragment")
|
return strings.HasPrefix(s, OpenLink)
|
||||||
for {
|
|
||||||
if strings.HasPrefix(l.input[l.pos:], CloseLink) {
|
|
||||||
return lexCloseLink
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if l.peek() == '^' {
|
func isCloseLink(s string) bool {
|
||||||
l.next()
|
return strings.HasPrefix(s, CloseLink)
|
||||||
l.emit(ItemFragment)
|
|
||||||
l.acceptRun("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789 -.,")
|
|
||||||
|
|
||||||
return lexInsideLink
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return lexInsideLink
|
func isAlias(s string) bool {
|
||||||
}
|
return strings.HasPrefix(s, Alias)
|
||||||
}
|
}
|
||||||
|
|
||||||
func lexAlias(l *Lexer) stateFn {
|
func isHeading(s string) bool {
|
||||||
// l.accept
|
return strings.HasPrefix(s, Heading)
|
||||||
log.Println("entering lexAlias")
|
|
||||||
for {
|
|
||||||
if strings.HasPrefix(l.input[l.pos:], CloseLink) {
|
|
||||||
return lexCloseLink
|
|
||||||
}
|
}
|
||||||
r := l.next()
|
|
||||||
|
|
||||||
}
|
func isBlockRef(s string) bool {
|
||||||
l.emit(ItemAlias)
|
return strings.HasPrefix(s, BlockRef)
|
||||||
|
|
||||||
return lexInsideLink
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func lexIdent(l *Lexer) stateFn {
|
func lexIdent(l *Lexer) stateFn {
|
||||||
for {
|
for {
|
||||||
if strings.HasPrefix(l.input[l.pos:], CloseLink) {
|
r := l.next()
|
||||||
|
s := l.input[l.pos:]
|
||||||
|
if r == '\\' { // i think this will handle escape characters?
|
||||||
|
break
|
||||||
|
}
|
||||||
|
switch {
|
||||||
|
case isBlockRef(s):
|
||||||
|
l.emit(ItemIdent)
|
||||||
|
return lexBlockRef
|
||||||
|
case isAlias(s):
|
||||||
|
l.emit(ItemIdent)
|
||||||
|
return lexAlias
|
||||||
|
case isCloseLink(s):
|
||||||
|
l.emit(ItemIdent)
|
||||||
return lexCloseLink
|
return lexCloseLink
|
||||||
|
case isHeading(s):
|
||||||
|
l.emit(ItemIdent)
|
||||||
|
return lexHeading
|
||||||
}
|
}
|
||||||
// r := l.next()
|
|
||||||
}
|
}
|
||||||
return lexInsideLink
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func lexInsideLink(l *Lexer) stateFn {
|
func lexHeading(l *Lexer) stateFn {
|
||||||
log.Println("entering lexInsideLink")
|
l.pos += len(Heading)
|
||||||
|
l.emit(ItemHeading)
|
||||||
|
|
||||||
for {
|
return lexIdent
|
||||||
if strings.HasPrefix(l.input[l.pos:], CloseLink) {
|
|
||||||
return lexCloseLink
|
|
||||||
}
|
}
|
||||||
|
|
||||||
r := l.next()
|
func lexBlockRef(l *Lexer) stateFn {
|
||||||
log.Printf("inspecting %q\n", string(r))
|
l.pos += len(BlockRef)
|
||||||
|
l.emit(ItemBlockRef)
|
||||||
|
|
||||||
switch {
|
return lexIdent
|
||||||
case r == EOF:
|
}
|
||||||
case r == '\n':
|
|
||||||
return l.errorf("unclosed link")
|
|
||||||
case r == '#':
|
|
||||||
l.emit(ItemText)
|
|
||||||
|
|
||||||
return lexFragment
|
func lexAlias(l *Lexer) stateFn {
|
||||||
case r == '|':
|
l.pos += len(Alias)
|
||||||
l.emit(ItemText)
|
l.emit(ItemAlias)
|
||||||
|
|
||||||
return lexAlias
|
return lexIdent
|
||||||
|
}
|
||||||
|
|
||||||
|
func lexText(l *Lexer) stateFn {
|
||||||
|
for {
|
||||||
|
if isOpenLink(l.input[l.pos:]) {
|
||||||
|
return lexOpenLink
|
||||||
|
}
|
||||||
|
r := l.next()
|
||||||
|
switch {
|
||||||
|
case r == EOF || r == '\n':
|
||||||
|
return l.errorf("wikilink terminated incorrectly")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func lexOpenLink(l *Lexer) stateFn {
|
func lexOpenLink(l *Lexer) stateFn {
|
||||||
log.Println("entering lexOpenLink")
|
|
||||||
l.pos += len(OpenLink)
|
l.pos += len(OpenLink)
|
||||||
l.emit(ItemOpenLink)
|
l.emit(ItemOpenLink)
|
||||||
|
|
||||||
return lexInsideLink
|
return lexIdent
|
||||||
}
|
}
|
||||||
|
|
||||||
func lexCloseLink(l *Lexer) stateFn {
|
func lexCloseLink(l *Lexer) stateFn {
|
||||||
log.Println("entering lexCloseLink")
|
|
||||||
l.pos += len(CloseLink)
|
l.pos += len(CloseLink)
|
||||||
l.emit(ItemCloseLink)
|
l.emit(ItemCloseLink)
|
||||||
|
|
||||||
return lexText
|
return lexText
|
||||||
}
|
}
|
||||||
|
|
||||||
func lexText(l *Lexer) stateFn {
|
/*
|
||||||
log.Println("entering lexText")
|
func lexInsideLink(l *Lexer) stateFn {
|
||||||
|
log.Println("entering lexInsideLink")
|
||||||
|
|
||||||
for {
|
for {
|
||||||
if strings.HasPrefix(l.input[l.pos:], OpenLink) {
|
if strings.HasPrefix(l.input[l.pos:], CloseLink) {
|
||||||
if l.pos > l.start {
|
return lexCloseLink
|
||||||
l.emit(ItemText)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return lexOpenLink
|
r := l.next()
|
||||||
}
|
log.Printf("inspecting %q\n", string(r))
|
||||||
|
|
||||||
if l.next() == EOF {
|
switch {
|
||||||
break
|
case r == EOF:
|
||||||
}
|
case r == '\n':
|
||||||
if l.pos > l.start {
|
return l.errorf("unclosed link")
|
||||||
|
case r == '#':
|
||||||
l.emit(ItemText)
|
l.emit(ItemText)
|
||||||
|
if l.peek() = '^' { return lexBlockRef }
|
||||||
|
|
||||||
|
return lexHeading
|
||||||
|
case r == '|':
|
||||||
|
l.emit(ItemText)
|
||||||
|
|
||||||
|
return lexAlias
|
||||||
}
|
}
|
||||||
l.emit(ItemEOF)
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
*/
|
||||||
|
Loading…
Reference in New Issue