|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
import { |
|
asciiAlpha, |
|
asciiAlphanumeric, |
|
asciiAtext, |
|
asciiControl |
|
} from 'micromark-util-character' |
|
import {codes} from 'micromark-util-symbol/codes.js' |
|
import {constants} from 'micromark-util-symbol/constants.js' |
|
import {types} from 'micromark-util-symbol/types.js' |
|
import {ok as assert} from 'uvu/assert' |
|
|
|
|
|
export const autolink = {name: 'autolink', tokenize: tokenizeAutolink} |
|
|
|
|
|
|
|
|
|
|
|
function tokenizeAutolink(effects, ok, nok) { |
|
let size = 0 |
|
|
|
return start |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function start(code) { |
|
assert(code === codes.lessThan, 'expected `<`') |
|
effects.enter(types.autolink) |
|
effects.enter(types.autolinkMarker) |
|
effects.consume(code) |
|
effects.exit(types.autolinkMarker) |
|
effects.enter(types.autolinkProtocol) |
|
return open |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function open(code) { |
|
if (asciiAlpha(code)) { |
|
effects.consume(code) |
|
return schemeOrEmailAtext |
|
} |
|
|
|
return emailAtext(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function schemeOrEmailAtext(code) { |
|
|
|
if ( |
|
code === codes.plusSign || |
|
code === codes.dash || |
|
code === codes.dot || |
|
asciiAlphanumeric(code) |
|
) { |
|
|
|
size = 1 |
|
return schemeInsideOrEmailAtext(code) |
|
} |
|
|
|
return emailAtext(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function schemeInsideOrEmailAtext(code) { |
|
if (code === codes.colon) { |
|
effects.consume(code) |
|
size = 0 |
|
return urlInside |
|
} |
|
|
|
|
|
if ( |
|
(code === codes.plusSign || |
|
code === codes.dash || |
|
code === codes.dot || |
|
asciiAlphanumeric(code)) && |
|
size++ < constants.autolinkSchemeSizeMax |
|
) { |
|
effects.consume(code) |
|
return schemeInsideOrEmailAtext |
|
} |
|
|
|
size = 0 |
|
return emailAtext(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function urlInside(code) { |
|
if (code === codes.greaterThan) { |
|
effects.exit(types.autolinkProtocol) |
|
effects.enter(types.autolinkMarker) |
|
effects.consume(code) |
|
effects.exit(types.autolinkMarker) |
|
effects.exit(types.autolink) |
|
return ok |
|
} |
|
|
|
|
|
if ( |
|
code === codes.eof || |
|
code === codes.space || |
|
code === codes.lessThan || |
|
asciiControl(code) |
|
) { |
|
return nok(code) |
|
} |
|
|
|
effects.consume(code) |
|
return urlInside |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function emailAtext(code) { |
|
if (code === codes.atSign) { |
|
effects.consume(code) |
|
return emailAtSignOrDot |
|
} |
|
|
|
if (asciiAtext(code)) { |
|
effects.consume(code) |
|
return emailAtext |
|
} |
|
|
|
return nok(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function emailAtSignOrDot(code) { |
|
return asciiAlphanumeric(code) ? emailLabel(code) : nok(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function emailLabel(code) { |
|
if (code === codes.dot) { |
|
effects.consume(code) |
|
size = 0 |
|
return emailAtSignOrDot |
|
} |
|
|
|
if (code === codes.greaterThan) { |
|
|
|
effects.exit(types.autolinkProtocol).type = types.autolinkEmail |
|
effects.enter(types.autolinkMarker) |
|
effects.consume(code) |
|
effects.exit(types.autolinkMarker) |
|
effects.exit(types.autolink) |
|
return ok |
|
} |
|
|
|
return emailValue(code) |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function emailValue(code) { |
|
|
|
if ( |
|
(code === codes.dash || asciiAlphanumeric(code)) && |
|
size++ < constants.autolinkDomainSizeMax |
|
) { |
|
const next = code === codes.dash ? emailValue : emailLabel |
|
effects.consume(code) |
|
return next |
|
} |
|
|
|
return nok(code) |
|
} |
|
} |
|
|