import std/options
import std/os
import std/streams
import std/tables
import config/chapath
import config/mailcap
import config/mimetypes
import config/toml
import io/urlfilter
import js/error
import js/javascript
import js/propertyenumlist
import js/regex
import loader/headers
import loader/loader
import types/cell
import types/color
import types/cookie
import types/opt
import types/referer
import types/urimethodmap
import types/url
import utils/mimeguess
import utils/twtstr
import chakasu/charset
type
ColorMode* = enum
MONOCHROME, ANSI, EIGHT_BIT, TRUE_COLOR
FormatMode* = set[FormatFlags]
ActionMap = object
t: Table[string, string]
StaticSiteConfig = object
url: Opt[string]
host: Opt[string]
rewrite_url: Opt[string]
cookie: Opt[bool]
third_party_cookie: seq[string]
share_cookie_jar: Opt[string]
referer_from*: Opt[bool]
scripting: Opt[bool]
document_charset: seq[Charset]
images: Opt[bool]
stylesheet: Opt[string]
proxy: Opt[string]
StaticOmniRule = object
match: string
substitute_url: string
SiteConfig* = object
url*: Opt[Regex]
host*: Opt[Regex]
rewrite_url*: (proc(s: URL): JSResult[URL])
cookie*: Opt[bool]
third_party_cookie*: seq[Regex]
share_cookie_jar*: Opt[string]
referer_from*: Opt[bool]
scripting*: Opt[bool]
document_charset*: seq[Charset]
images*: Opt[bool]
stylesheet*: Opt[string]
proxy*: Opt[URL]
OmniRule* = object
match*: Regex
substitute_url*: (proc(s: string): JSResult[string])
StartConfig = object
visual_home* {.jsgetset.}: string
startup_script* {.jsgetset.}: string
headless* {.jsgetset.}: bool
CSSConfig = object
stylesheet* {.jsgetset.}: string
SearchConfig = object
wrap* {.jsgetset.}: bool
EncodingConfig = object
display_charset* {.jsgetset.}: Opt[Charset]
document_charset* {.jsgetset.}: seq[Charset]
ExternalConfig = object
tmpdir* {.jsgetset.}: ChaPath
editor* {.jsgetset.}: string
mailcap* {.jsgetset.}: seq[ChaPath]
mime_types* {.jsgetset.}: seq[ChaPath]
cgi_dir* {.jsgetset.}: seq[ChaPath]
urimethodmap* {.jsgetset.}: seq[ChaPath]
w3m_cgi_compat* {.jsgetset.}: bool
InputConfig = object
vi_numeric_prefix* {.jsgetset.}: bool
NetworkConfig = object
max_redirect* {.jsgetset.}: int32
prepend_https* {.jsgetset.}: bool
prepend_scheme* {.jsgetset.}: string
proxy* {.jsgetset.}: Opt[string]
default_headers* {.jsgetset.}: Table[string, string]
DisplayConfig = object
color_mode* {.jsgetset.}: Opt[ColorMode]
format_mode*: Opt[FormatMode] #TODO getset
no_format_mode*: FormatMode #TODO getset
emulate_overline* {.jsgetset.}: bool
alt_screen* {.jsgetset.}: Opt[bool]
highlight_color* {.jsgetset.}: RGBAColor
highlight_marks* {.jsgetset.}: bool
double_width_ambiguous* {.jsgetset.}: bool
minimum_contrast* {.jsgetset.}: int32
force_clear* {.jsgetset.}: bool
set_title* {.jsgetset.}: bool
default_background_color* {.jsgetset.}: RGBColor
default_foreground_color* {.jsgetset.}: RGBColor
Config* = ref ConfigObj
ConfigObj* = object
configdir {.jsget.}: string
`include` {.jsget.}: seq[ChaPath]
start* {.jsget.}: StartConfig
search* {.jsget.}: SearchConfig
css* {.jsget.}: CSSConfig
encoding* {.jsget.}: EncodingConfig
external* {.jsget.}: ExternalConfig
network* {.jsget.}: NetworkConfig
input* {.jsget.}: InputConfig
display* {.jsget.}: DisplayConfig
#TODO getset
siteconf: seq[StaticSiteConfig]
omnirule: seq[StaticOmniRule]
page* {.jsget.}: ActionMap
line* {.jsget.}: ActionMap
BufferConfig* = object
userstyle*: string
referer_from*: bool
referrerpolicy*: ReferrerPolicy
scripting*: bool
charsets*: seq[Charset]
images*: bool
loaderConfig*: LoaderConfig
mimeTypes*: MimeTypes
cgiDir*: seq[string]
ForkServerConfig* = object
tmpdir*: string
ambiguous_double*: bool
jsDestructor(ActionMap)
jsDestructor(StartConfig)
jsDestructor(CSSConfig)
jsDestructor(SearchConfig)
jsDestructor(EncodingConfig)
jsDestructor(ExternalConfig)
jsDestructor(NetworkConfig)
jsDestructor(DisplayConfig)
jsDestructor(Config)
proc `[]=`(a: var ActionMap, b, c: string) = a.t[b] = c
proc `[]`*(a: ActionMap, b: string): string = a.t[b]
proc contains*(a: ActionMap, b: string): bool = b in a.t
proc getOrDefault(a: ActionMap, b: string): string = a.t.getOrDefault(b)
proc hasKeyOrPut(a: var ActionMap, b, c: string): bool = a.t.hasKeyOrPut(b, c)
func getRealKey(key: string): string
proc getter(a: ptr ActionMap, s: string): Opt[string] {.jsgetprop.} =
a.t.withValue(s, p):
return opt(p[])
proc setter(a: ptr ActionMap, k, v: string) {.jssetprop.} =
let k = getRealKey(k)
if k == "":
return
a[][k] = v
var teststr = k
teststr.setLen(teststr.high)
for i in countdown(k.high, 0):
if teststr notin a[]:
a[][teststr] = "client.feedNext()"
teststr.setLen(i)
proc delete(a: ptr ActionMap, k: string): bool {.jsdelprop.} =
let k = getRealKey(k)
let ina = k in a[]
a[].t.del(k)
return ina
func names(ctx: JSContext, a: ptr ActionMap): JSPropertyEnumList
{.jspropnames.} =
let L = uint32(a[].t.len)
var list = newJSPropertyEnumList(ctx, L)
for key in a[].t.keys:
list.add(key)
return list
proc bindPagerKey(config: Config, key, action: string) {.jsfunc.} =
(addr config.page).setter(key, action)
proc bindLineKey(config: Config, key, action: string) {.jsfunc.} =
(addr config.line).setter(key, action)
proc hasprop(a: ptr ActionMap, s: string): bool {.jshasprop.} =
return s in a[]
func getProxy*(config: Config): URL =
if config.network.proxy.isSome:
let s = config.network.proxy.get
let x = parseURL(s)
if x.isSome:
return x.get
else:
raise newException(Defect, "Invalid proxy URL: " & s)
return nil
func getDefaultHeaders*(config: Config): Headers =
return newHeaders(config.network.default_headers)
proc getBufferConfig*(config: Config, location: URL, cookiejar: CookieJar,
headers: Headers, referer_from, scripting: bool, charsets: seq[Charset],
images: bool, userstyle: string, proxy: URL, mimeTypes: MimeTypes,
urimethodmap: URIMethodMap, cgiDir: seq[string]): BufferConfig =
let filter = newURLFilter(
scheme = some(location.scheme),
allowschemes = @["data"],
default = true
)
return BufferConfig(
userstyle: userstyle,
referer_from: referer_from,
scripting: scripting,
charsets: charsets,
images: images,
mimeTypes: mimeTypes,
loaderConfig: LoaderConfig(
defaultHeaders: headers,
filter: filter,
cookiejar: cookiejar,
proxy: proxy,
cgiDir: cgiDir,
urimethodmap: urimethodmap,
w3mCGICompat: config.external.w3m_cgi_compat,
libexecPath: ChaPath("${%CHA_LIBEXEC_DIR}").unquote().get
)
)
proc getSiteConfig*(config: Config, jsctx: JSContext): seq[SiteConfig] =
for sc in config.siteconf:
var conf = SiteConfig(
cookie: sc.cookie,
scripting: sc.scripting,
share_cookie_jar: sc.share_cookie_jar,
referer_from: sc.referer_from,
document_charset: sc.document_charset,
images: sc.images
)
if sc.url.isSome:
conf.url = opt(compileMatchRegex(sc.url.get))
elif sc.host.isSome:
conf.host = opt(compileMatchRegex(sc.host.get))
for rule in sc.third_party_cookie:
conf.third_party_cookie.add(compileMatchRegex(rule).get)
if sc.rewrite_url.isSome:
let fun = jsctx.eval(sc.rewrite_url.get, "<siteconf>",
JS_EVAL_TYPE_GLOBAL)
conf.rewrite_url = getJSFunction[URL, URL](jsctx, fun)
if sc.proxy.isSome:
let x = parseURL(sc.proxy.get)
if x.isNone:
raise newException(Defect, "invalid URL: " & sc.proxy.get)
conf.proxy = opt(x.get)
result.add(conf)
proc getOmniRules*(config: Config, jsctx: JSContext): seq[OmniRule] =
for rule in config.omnirule:
let re = compileMatchRegex(rule.match)
var conf = OmniRule(
match: re.get
)
let fun = jsctx.eval(rule.substitute_url, "<siteconf>", JS_EVAL_TYPE_GLOBAL)
conf.substitute_url = getJSFunction[string, string](jsctx, fun)
result.add(conf)
func getRealKey(key: string): string =
var realk: string
var control = 0
var meta = 0
var skip = false
for c in key:
if c == '\\':
skip = true
elif skip:
realk &= c
skip = false
elif c == 'M' and meta == 0:
inc meta
elif c == 'C' and control == 0:
inc control
elif c == '-' and control == 1:
inc control
elif c == '-' and meta == 1:
inc meta
elif meta == 1:
realk &= 'M' & c
meta = 0
elif control == 1:
realk &= 'C' & c
control = 0
else:
if meta == 2:
realk &= '\e'
meta = 0
if control == 2:
realk &= getControlChar(c)
control = 0
else:
realk &= c
if control == 1:
realk &= 'C'
if meta == 1:
realk &= 'M'
if skip:
realk &= '\\'
return realk
proc openFileExpand(dir: string, file: ChaPath): FileStream =
let file0 = file.unquote()
if file0.isNone:
raise newException(ValueError, file0.error)
let file = file0.get
if file.len == 0:
return nil
if file[0] == '/':
return newFileStream(file)
else:
return newFileStream(dir / file)
proc readUserStylesheet(dir, file: string): string =
let s = openFileExpand(dir, ChaPath(file))
if s != nil:
result = s.readAll()
s.close()
# The overall configuration will be obtained through the virtual concatenation
# of several individual configuration files known as mailcap files.
proc getMailcap*(config: Config): tuple[mailcap: Mailcap, errs: seq[string]] =
let configDir = getConfigDir() / "chawan" #TODO store this in config?
const gopherPath0 = ChaPath("${%CHA_LIBEXEC_DIR}/gopher2html -u \\$MAILCAP_URL")
let gopherPath = gopherPath0.unquote().get
const geminiPath0 = ChaPath("${%CHA_LIBEXEC_DIR}/gmi2html")
let geminiPath = geminiPath0.unquote().get
const mdPath0 = ChaPath("${%CHA_LIBEXEC_DIR}/md2html")
let mdPath = mdPath0.unquote().get
var mailcap: Mailcap = @[]
var errs: seq[string]
var found = false
for p in config.external.mailcap:
let f = openFileExpand(configDir, p)
if f != nil:
let res = parseMailcap(f)
if res.isSome:
mailcap.add(res.get)
else:
errs.add(res.error)
found = true
mailcap.add(MailcapEntry(
mt: "text",
subt: "gopher",
cmd: gopherPath,
flags: {HTMLOUTPUT}
))
mailcap.add(MailcapEntry(
mt: "text",
subt: "gemini",
cmd: geminiPath,
flags: {HTMLOUTPUT}
))
mailcap.add(MailcapEntry(
mt: "text",
subt: "markdown",
cmd: mdPath,
flags: {HTMLOUTPUT}
))
if not found:
mailcap.insert(MailcapEntry(
mt: "*",
subt: "*",
cmd: "xdg-open '%s'"
), 0)
return (DefaultMailcap, errs)
return (mailcap, errs)
# We try to source mime types declared in config.
# If none of these files can be found, fall back to DefaultGuess.
#TODO some error handling would be nice, to at least show a warning to
# the user. Not sure how this could be done, though.
proc getMimeTypes*(config: Config): MimeTypes =
if config.external.mime_types.len == 0:
return DefaultGuess
var mimeTypes: MimeTypes
let configDir = getConfigDir() / "chawan" #TODO store this in config?
var found = false
for p in config.external.mime_types:
let f = openFileExpand(configDir, p)
if f != nil:
mimeTypes.parseMimeTypes(f)
found = true
if not found:
return DefaultGuess
return mimeTypes
const DefaultURIMethodMap = parseURIMethodMap(staticRead"res/urimethodmap")
proc getURIMethodMap*(config: Config): URIMethodMap =
let configDir = getConfigDir() / "chawan" #TODO store this in config?
var urimethodmap: URIMethodMap
for p in config.external.urimethodmap:
let f = openFileExpand(configDir, p)
if f != nil:
urimethodmap.parseURIMethodMap(f.readAll())
urimethodmap.append(DefaultURIMethodMap)
return urimethodmap
proc getForkServerConfig*(config: Config): ForkServerConfig =
let tmpdir0 = config.external.tmpdir.unquote()
if tmpdir0.isNone:
raise newException(ValueError, tmpdir0.error)
return ForkServerConfig(
tmpdir: tmpdir0.get,
ambiguous_double: config.display.double_width_ambiguous
)
proc parseConfig(config: Config, dir: string, stream: Stream, name = "<input>",
laxnames = false)
proc parseConfig*(config: Config, dir: string, s: string, name = "<input>",
laxnames = false)
proc loadConfig*(config: Config, s: string) {.jsfunc.} =
let s = if s.len > 0 and s[0] == '/':
s
else:
getCurrentDir() / s
if not fileExists(s): return
config.parseConfig(parentDir(s), newFileStream(s))
proc parseConfigValue(x: var object, v: TomlValue, k: string)
proc parseConfigValue(x: var bool, v: TomlValue, k: string)
proc parseConfigValue(x: var string, v: TomlValue, k: string)
proc parseConfigValue(x: var ChaPath, v: TomlValue, k: string)
proc parseConfigValue[T](x: var seq[T], v: TomlValue, k: string)
proc parseConfigValue(x: var Charset, v: TomlValue, k: string)
proc parseConfigValue(x: var int32, v: TomlValue, k: string)
proc parseConfigValue(x: var int64, v: TomlValue, k: string)
proc parseConfigValue(x: var Opt[ColorMode], v: TomlValue, k: string)
proc parseConfigValue(x: var Opt[FormatMode], v: TomlValue, k: string)
proc parseConfigValue(x: var FormatMode, v: TomlValue, k: string)
proc parseConfigValue(x: var RGBAColor, v: TomlValue, k: string)
proc parseConfigValue(x: var RGBColor, v: TomlValue, k: string)
proc parseConfigValue[T](x: var Opt[T], v: TomlValue, k: string)
proc parseConfigValue(x: var ActionMap, v: TomlValue, k: string)
proc parseConfigValue(x: var CSSConfig, v: TomlValue, k: string)
proc parseConfigValue[U, V](x: var Table[U, V], v: TomlValue, k: string)
proc typeCheck(v: TomlValue, vt: ValueType, k: string) =
if v.vt != vt:
raise newException(ValueError, "invalid type for key " & k &
" (got " & $v.vt & ", expected " & $vt & ")")
proc typeCheck(v: TomlValue, vt: set[ValueType], k: string) =
if v.vt notin vt:
raise newException(ValueError, "invalid type for key " & k &
" (got " & $v.vt & ", expected " & $vt & ")")
proc parseConfigValue(x: var object, v: TomlValue, k: string) =
typeCheck(v, VALUE_TABLE, k)
for fk, fv in x.fieldPairs:
let kebabk = snakeToKebabCase(fk)
if kebabk in v:
let kkk = if k != "":
k & "." & fk
else:
fk
parseConfigValue(fv, v[kebabk], kkk)
proc parseConfigValue[U, V](x: var Table[U, V], v: TomlValue, k: string) =
typeCheck(v, VALUE_TABLE, k)
x.clear()
for kk, vv in v:
var y: V
let kkk = k & "[" & kk & "]"
parseConfigValue(y, vv, kkk)
x[kk] = y
proc parseConfigValue(x: var bool, v: TomlValue, k: string) =
typeCheck(v, VALUE_BOOLEAN, k)
x = v.b
proc parseConfigValue(x: var string, v: TomlValue, k: string) =
typeCheck(v, VALUE_STRING, k)
x = v.s
proc parseConfigValue(x: var ChaPath, v: TomlValue, k: string) =
typeCheck(v, VALUE_STRING, k)
x = ChaPath(v.s)
proc parseConfigValue[T](x: var seq[T], v: TomlValue, k: string) =
typeCheck(v, {VALUE_STRING, VALUE_ARRAY}, k)
if v.vt != VALUE_ARRAY:
var y: T
parseConfigValue(y, v, k)
x = @[y]
else:
if not v.ad:
x.setLen(0)
for i in 0 ..< v.a.len:
var y: T
parseConfigValue(y, v.a[i], k & "[" & $i & "]")
x.add(y)
proc parseConfigValue(x: var Charset, v: TomlValue, k: string) =
typeCheck(v, VALUE_STRING, k)
x = getCharset(v.s)
if x == CHARSET_UNKNOWN:
raise newException(ValueError, "unknown charset '" & v.s & "' for key " &
k)
proc parseConfigValue(x: var int32, v: TomlValue, k: string) =
typeCheck(v, VALUE_INTEGER, k)
x = int32(v.i)
proc parseConfigValue(x: var int64, v: TomlValue, k: string) =
typeCheck(v, VALUE_INTEGER, k)
x = v.i
proc parseConfigValue(x: var Opt[ColorMode], v: TomlValue, k: string) =
typeCheck(v, VALUE_STRING, k)
case v.s
of "auto": x.err()
of "monochrome": x.ok(MONOCHROME)
of "ansi": x.ok(ANSI)
of "8bit", "eight-bit": x.ok(EIGHT_BIT)
of "24bit", "true-color": x.ok(TRUE_COLOR)
else:
raise newException(ValueError, "unknown color mode '" & v.s &
"' for key " & k)
proc parseConfigValue(x: var Opt[FormatMode], v: TomlValue, k: string) =
typeCheck(v, {VALUE_STRING, VALUE_ARRAY}, k)
if v.vt == VALUE_STRING and v.s == "auto":
x.err()
else:
var y: FormatMode
parseConfigValue(y, v, k)
x.ok(y)
proc parseConfigValue(x: var FormatMode, v: TomlValue, k: string) =
typeCheck(v, VALUE_ARRAY, k)
for i in 0 ..< v.a.len:
let s = v.a[i].s
let kk = k & "[" & $i & "]"
case s
of "bold": x.incl(FLAG_BOLD)
of "italic": x.incl(FLAG_ITALIC)
of "underline": x.incl(FLAG_UNDERLINE)
of "reverse": x.incl(FLAG_REVERSE)
of "strike": x.incl(FLAG_STRIKE)
of "overline": x.incl(FLAG_OVERLINE)
of "blink": x.incl(FLAG_BLINK)
else:
raise newException(ValueError, "unknown format mode '" & s &
"' for key " & kk)
proc parseConfigValue(x: var RGBAColor, v: TomlValue, k: string) =
typeCheck(v, VALUE_STRING, k)
let c = parseRGBAColor(v.s)
if c.isNone:
raise newException(ValueError, "invalid color '" & v.s &
"' for key " & k)
x = c.get
proc parseConfigValue(x: var RGBColor, v: TomlValue, k: string) =
typeCheck(v, VALUE_STRING, k)
let c = parseLegacyColor(v.s)
if c.isNone:
raise newException(ValueError, "invalid color '" & v.s &
"' for key " & k)
x = c.get
proc parseConfigValue[T](x: var Opt[T], v: TomlValue, k: string) =
if v.vt == VALUE_STRING and v.s == "auto":
x.err()
else:
var y: T
parseConfigValue(y, v, k)
x.ok(y)
proc parseConfigValue(x: var ActionMap, v: TomlValue, k: string) =
typeCheck(v, VALUE_TABLE, k)
for kk, vv in v:
typeCheck(vv, VALUE_STRING, k & "[" & kk & "]")
let rk = getRealKey(kk)
var buf: string
for i in 0 ..< rk.high:
buf &= rk[i]
discard x.hasKeyOrPut(buf, "client.feedNext()")
x[rk] = vv.s
var gdir {.compileTime.}: string
proc parseConfigValue(x: var CSSConfig, v: TomlValue, k: string) =
typeCheck(v, VALUE_TABLE, k)
let dir = gdir
for kk, vv in v:
let kkk = if k != "":
k & "." & kk
else:
kk
case kk
of "include":
typeCheck(vv, {VALUE_STRING, VALUE_ARRAY}, kkk)
case vv.vt
of VALUE_STRING:
x.stylesheet &= readUserStylesheet(dir, vv.s)
of VALUE_ARRAY:
for child in vv.a:
x.stylesheet &= readUserStylesheet(dir, vv.s)
else: discard
of "inline":
typeCheck(vv, VALUE_STRING, kkk)
x.stylesheet &= vv.s
proc parseConfig(config: Config, dir: string, t: TomlValue) =
gdir = dir
parseConfigValue(config[], t, "")
while config.`include`.len > 0:
#TODO: warn about recursive includes
var includes = config.`include`
config.`include`.setLen(0)
for s in includes:
when nimvm:
config.parseConfig(dir, staticRead(dir / string(s)))
else:
config.parseConfig(dir, openFileExpand(dir, s))
config.configdir = dir
#TODO: for omnirule/siteconf, check if substitution rules are specified?
proc parseConfig(config: Config, dir: string, stream: Stream, name = "<input>",
laxnames = false) =
let toml = parseToml(stream, dir / name, laxnames)
if toml.isOk:
config.parseConfig(dir, toml.get)
else:
when nimvm:
echo "Fatal error: Failed to parse config"
echo toml.error
else:
stderr.write("Fatal error: Failed to parse config\n")
stderr.write(toml.error & '\n')
quit(1)
proc parseConfig*(config: Config, dir: string, s: string, name = "<input>",
laxnames = false) =
config.parseConfig(dir, newStringStream(s), name, laxnames)
proc staticReadConfig(): ConfigObj =
var config = Config()
config.parseConfig("res", staticRead"res/config.toml", "config.toml")
return config[]
const defaultConfig = staticReadConfig()
proc readConfig(config: Config, dir, name: string) =
let fs = if name.len > 0 and name[0] == '/':
newFileStream(name)
else:
newFileStream(dir / name)
if fs != nil:
config.parseConfig(dir, fs)
proc getNormalAction*(config: Config, s: string): string =
return config.page.getOrDefault(s)
proc getLinedAction*(config: Config, s: string): string =
return config.line.getOrDefault(s)
proc readConfig*(pathOverride: Option[string]): Config =
result = Config()
result[] = defaultConfig
if pathOverride.isNone:
when defined(debug):
result.readConfig(getCurrentDir() / "res", "config.toml")
result.readConfig(getConfigDir() / "chawan", "config.toml")
else:
result.readConfig(getCurrentDir(), pathOverride.get)
proc addConfigModule*(ctx: JSContext) =
ctx.registerType(ActionMap)
ctx.registerType(StartConfig)
ctx.registerType(CSSConfig)
ctx.registerType(SearchConfig)
ctx.registerType(EncodingConfig)
ctx.registerType(ExternalConfig)
ctx.registerType(NetworkConfig)
ctx.registerType(DisplayConfig)
ctx.registerType(Config)