Merge branch 'devel' into sighashes

This commit is contained in:
Andreas Rumpf
2016-12-06 10:08:38 +01:00
10 changed files with 242 additions and 178 deletions

View File

@@ -16,7 +16,7 @@ type
{.deprecated: [Tcflag: Cflag].}
const
NCCS* = 32
NCCS* = when defined(macosx): 20 else: 32
type
Termios* {.importc: "struct termios", header: "<termios.h>".} = object
@@ -28,174 +28,136 @@ type
# cc characters
const
VINTR* = 0
VQUIT* = 1
VERASE* = 2
VKILL* = 3
VEOF* = 4
VTIME* = 5
VMIN* = 6
VSWTC* = 7
VSTART* = 8
VSTOP* = 9
VSUSP* = 10
VEOL* = 11
VREPRINT* = 12
VDISCARD* = 13
VWERASE* = 14
VLNEXT* = 15
VEOL2* = 16
var
VINTR* {.importc, header: "<termios.h>".}: cint
VQUIT* {.importc, header: "<termios.h>".}: cint
VERASE* {.importc, header: "<termios.h>".}: cint
VKILL* {.importc, header: "<termios.h>".}: cint
VEOF* {.importc, header: "<termios.h>".}: cint
VTIME* {.importc, header: "<termios.h>".}: cint
VMIN* {.importc, header: "<termios.h>".}: cint
VSTART* {.importc, header: "<termios.h>".}: cint
VSTOP* {.importc, header: "<termios.h>".}: cint
VSUSP* {.importc, header: "<termios.h>".}: cint
VEOL* {.importc, header: "<termios.h>".}: cint
# iflag bits
const
IGNBRK* = 1
BRKINT* = 2
IGNPAR* = 4
PARMRK* = 10
INPCK* = 20
ISTRIP* = 40
INLCR* = 100
IGNCR* = 200
ICRNL* = 400
IUCLC* = 1000
IXON* = 2000
IXANY* = 4000
IXOFF* = 10000
IMAXBEL* = 20000
IUTF8* = 40000
var
IGNBRK* {.importc, header: "<termios.h>".}: Cflag
BRKINT* {.importc, header: "<termios.h>".}: Cflag
IGNPAR* {.importc, header: "<termios.h>".}: Cflag
PARMRK* {.importc, header: "<termios.h>".}: Cflag
INPCK* {.importc, header: "<termios.h>".}: Cflag
ISTRIP* {.importc, header: "<termios.h>".}: Cflag
INLCR* {.importc, header: "<termios.h>".}: Cflag
IGNCR* {.importc, header: "<termios.h>".}: Cflag
ICRNL* {.importc, header: "<termios.h>".}: Cflag
IUCLC* {.importc, header: "<termios.h>".}: Cflag
IXON* {.importc, header: "<termios.h>".}: Cflag
IXANY* {.importc, header: "<termios.h>".}: Cflag
IXOFF* {.importc, header: "<termios.h>".}: Cflag
# oflag bits
const
OPOST* = 1
OLCUC* = 2
ONLCR* = 4
OCRNL* = 10
ONOCR* = 20
ONLRET* = 40
OFILL* = 100
OFDEL* = 200
NLDLY* = 400
NL0* = 0
NL1* = 400
CRDLY* = 3000
CR0* = 0
CR1* = 1000
CR2* = 2000
CR3* = 3000
TABDLY* = 14000
TAB0* = 0
TAB1* = 4000
TAB2* = 10000
TAB3* = 14000
BSDLY* = 20000
BS0* = 0
BS1* = 20000
FFDLY* = 0o000000100000
FF0* = 0
FF1* = 0o000000100000
VTDLY* = 40000
VT0* = 0
VT1* = 40000
XTABS* = 14000
var
OPOST* {.importc, header: "<termios.h>".}: Cflag
ONLCR* {.importc, header: "<termios.h>".}: Cflag
OCRNL* {.importc, header: "<termios.h>".}: Cflag
ONOCR* {.importc, header: "<termios.h>".}: Cflag
ONLRET* {.importc, header: "<termios.h>".}: Cflag
OFILL* {.importc, header: "<termios.h>".}: Cflag
OFDEL* {.importc, header: "<termios.h>".}: Cflag
NLDLY* {.importc, header: "<termios.h>".}: Cflag
NL0* {.importc, header: "<termios.h>".}: Cflag
NL1* {.importc, header: "<termios.h>".}: Cflag
CRDLY* {.importc, header: "<termios.h>".}: Cflag
CR0* {.importc, header: "<termios.h>".}: Cflag
CR1* {.importc, header: "<termios.h>".}: Cflag
CR2* {.importc, header: "<termios.h>".}: Cflag
CR3* {.importc, header: "<termios.h>".}: Cflag
TABDLY* {.importc, header: "<termios.h>".}: Cflag
TAB0* {.importc, header: "<termios.h>".}: Cflag
TAB1* {.importc, header: "<termios.h>".}: Cflag
TAB2* {.importc, header: "<termios.h>".}: Cflag
TAB3* {.importc, header: "<termios.h>".}: Cflag
BSDLY* {.importc, header: "<termios.h>".}: Cflag
BS0* {.importc, header: "<termios.h>".}: Cflag
BS1* {.importc, header: "<termios.h>".}: Cflag
FFDLY* {.importc, header: "<termios.h>".}: Cflag
FF0* {.importc, header: "<termios.h>".}: Cflag
FF1* {.importc, header: "<termios.h>".}: Cflag
VTDLY* {.importc, header: "<termios.h>".}: Cflag
VT0* {.importc, header: "<termios.h>".}: Cflag
VT1* {.importc, header: "<termios.h>".}: Cflag
# cflag bit meaning
const
CBAUD* = 10017
B0* = 0
B50* = 1
B75* = 2
B110* = 3
B134* = 4
B150* = 5
B200* = 6
B300* = 7
B600* = 10
B1200* = 11
B1800* = 12
B2400* = 13
B4800* = 14
B9600* = 15
B19200* = 16
B38400* = 17
EXTA* = B19200
EXTB* = B38400
CSIZE* = 60
CS5* = 0
CS6* = 20
CS7* = 40
CS8* = 60
CSTOPB* = 100
CREAD* = 200
PARENB* = 400
PARODD* = 1000
HUPCL* = 2000
CLOCAL* = 4000
CBAUDEX* = 10000
B57600* = 10001
B115200* = 10002
B230400* = 10003
B460800* = 10004
B500000* = 10005
B576000* = 10006
B921600* = 10007
B1000000* = 10010
B1152000* = 10011
B1500000* = 10012
B2000000* = 10013
B2500000* = 10014
B3000000* = 10015
B3500000* = 10016
B4000000* = 10017
MAX_BAUD* = B4000000
CIBAUD* = 2003600000
CMSPAR* = 0o010000000000
CRTSCTS* = 0o020000000000
var
B0* {.importc, header: "<termios.h>".}: Speed
B50* {.importc, header: "<termios.h>".}: Speed
B75* {.importc, header: "<termios.h>".}: Speed
B110* {.importc, header: "<termios.h>".}: Speed
B134* {.importc, header: "<termios.h>".}: Speed
B150* {.importc, header: "<termios.h>".}: Speed
B200* {.importc, header: "<termios.h>".}: Speed
B300* {.importc, header: "<termios.h>".}: Speed
B600* {.importc, header: "<termios.h>".}: Speed
B1200* {.importc, header: "<termios.h>".}: Speed
B1800* {.importc, header: "<termios.h>".}: Speed
B2400* {.importc, header: "<termios.h>".}: Speed
B4800* {.importc, header: "<termios.h>".}: Speed
B9600* {.importc, header: "<termios.h>".}: Speed
B19200* {.importc, header: "<termios.h>".}: Speed
B38400* {.importc, header: "<termios.h>".}: Speed
EXTA* {.importc, header: "<termios.h>".}: Speed
EXTB* {.importc, header: "<termios.h>".}: Speed
CSIZE* {.importc, header: "<termios.h>".}: Cflag
CS5* {.importc, header: "<termios.h>".}: Cflag
CS6* {.importc, header: "<termios.h>".}: Cflag
CS7* {.importc, header: "<termios.h>".}: Cflag
CS8* {.importc, header: "<termios.h>".}: Cflag
CSTOPB* {.importc, header: "<termios.h>".}: Cflag
CREAD* {.importc, header: "<termios.h>".}: Cflag
PARENB* {.importc, header: "<termios.h>".}: Cflag
PARODD* {.importc, header: "<termios.h>".}: Cflag
HUPCL* {.importc, header: "<termios.h>".}: Cflag
CLOCAL* {.importc, header: "<termios.h>".}: Cflag
# lflag bits
const
ISIG* = 1
ICANON* = 2
XCASE* = 4
ECHO* = 10
ECHOE* = 20
ECHOK* = 40
ECHONL* = 100
NOFLSH* = 200
TOSTOP* = 400
ECHOCTL* = 1000
ECHOPRT* = 2000
ECHOKE* = 4000
FLUSHO* = 10000
PENDIN* = 40000
IEXTEN* = 0o000000100000
EXTPROC* = 0o000000200000
var
ISIG* {.importc, header: "<termios.h>".}: Cflag
ICANON* {.importc, header: "<termios.h>".}: Cflag
ECHO* {.importc, header: "<termios.h>".}: Cflag
ECHOE* {.importc, header: "<termios.h>".}: Cflag
ECHOK* {.importc, header: "<termios.h>".}: Cflag
ECHONL* {.importc, header: "<termios.h>".}: Cflag
NOFLSH* {.importc, header: "<termios.h>".}: Cflag
TOSTOP* {.importc, header: "<termios.h>".}: Cflag
IEXTEN* {.importc, header: "<termios.h>".}: Cflag
# tcflow() and TCXONC use these
const
TCOOFF* = 0
TCOON* = 1
TCIOFF* = 2
TCION* = 3
var
TCOOFF* {.importc, header: "<termios.h>".}: cint
TCOON* {.importc, header: "<termios.h>".}: cint
TCIOFF* {.importc, header: "<termios.h>".}: cint
TCION* {.importc, header: "<termios.h>".}: cint
# tcflush() and TCFLSH use these
const
TCIFLUSH* = 0
TCOFLUSH* = 1
TCIOFLUSH* = 2
var
TCIFLUSH* {.importc, header: "<termios.h>".}: cint
TCOFLUSH* {.importc, header: "<termios.h>".}: cint
TCIOFLUSH* {.importc, header: "<termios.h>".}: cint
# tcsetattr uses these
const
TCSANOW* = 0
TCSADRAIN* = 1
TCSAFLUSH* = 2
var
TCSANOW* {.importc, header: "<termios.h>".}: cint
TCSADRAIN* {.importc, header: "<termios.h>".}: cint
TCSAFLUSH* {.importc, header: "<termios.h>".}: cint
# Compare a character C to a value VAL from the `cc' array in a
# `struct termios'. If VAL is _POSIX_VDISABLE, no character can match it.
@@ -221,10 +183,6 @@ proc cfSetIspeed*(termios: ptr Termios; speed: Speed): cint {.
importc: "cfsetispeed", header: "<termios.h>".}
# Set both the input and output baud rates in *TERMIOS_OP to SPEED.
proc cfSetSpeed*(termios: ptr Termios; speed: Speed): cint {.
importc: "cfsetspeed", header: "<termios.h>".}
# Put the state of FD into *TERMIOS_P.
proc tcGetAttr*(fd: cint; termios: ptr Termios): cint {.
importc: "tcgetattr", header: "<termios.h>".}
# Set the state of FD to *TERMIOS_P.
@@ -234,10 +192,6 @@ proc tcSetAttr*(fd: cint; optional_actions: cint; termios: ptr Termios): cint {.
importc: "tcsetattr", header: "<termios.h>".}
# Set *TERMIOS_P to indicate raw mode.
proc cfMakeRaw*(termios: ptr Termios) {.importc: "cfmakeraw",
header: "<termios.h>".}
# Send zero bits on FD.
proc tcSendBreak*(fd: cint; duration: cint): cint {.importc: "tcsendbreak",
header: "<termios.h>".}
# Wait for pending output to be written on FD.
@@ -258,13 +212,10 @@ proc tcFlow*(fd: cint; action: cint): cint {.importc: "tcflow",
header: "<termios.h>".}
# Get process group ID for session leader for controlling terminal FD.
proc tcGetSid*(fd: cint): Pid {.importc: "tcgetsid", header: "<termios.h>".}
# Window size ioctl. Should work on on any Unix that xterm has been ported to.
var TIOCGWINSZ*{.importc, header: "<sys/ioctl.h>".}: culong
type IOctl_WinSize* {.importc: "struct winsize", header: "<sys/ioctl.h>",
final, pure.} = object
type IOctl_WinSize* = object
ws_row*, ws_col*, ws_xpixel*, ws_ypixel*: cushort
proc ioctl*(fd: cint, request: culong, reply: ptr IOctl_WinSize): int {.

View File

@@ -116,16 +116,15 @@ template hasKeyOrPutImpl(enlarge) {.dirty.} =
maybeRehashPutImpl(enlarge)
else: result = true
proc default[T](t: typedesc[T]): T {.inline.} = discard
template default[T](t: typedesc[T]): T =
var v: T
v
template delImpl() {.dirty.} =
var hc: Hash
var i = rawGet(t, key, hc)
let msk = maxHash(t)
if i >= 0:
t.data[i].hcode = 0
t.data[i].key = default(type(t.data[i].key))
t.data[i].val = default(type(t.data[i].val))
dec(t.counter)
block outer:
while true: # KnuthV3 Algo6.4R adapted for i=i+1 instead of i=i-1

View File

@@ -154,6 +154,10 @@ proc add*(headers: HttpHeaders, key, value: string) =
else:
headers.table[key.toLowerAscii].add(value)
proc del*(headers: HttpHeaders, key: string) =
## Delete the header entries associated with ``key``
headers.table.del(key.toLowerAscii)
iterator pairs*(headers: HttpHeaders): tuple[key, value: string] =
## Yields each key, value pair.
for k, v in headers.table:

View File

@@ -83,6 +83,13 @@ when defined(windows):
return int(csbi.srWindow.Right - csbi.srWindow.Left + 1)
return 0
proc terminalHeightIoctl*(handles: openArray[Handle]): int =
var csbi: CONSOLE_SCREEN_BUFFER_INFO
for h in handles:
if getConsoleScreenBufferInfo(h, addr csbi) != 0:
return int(csbi.srWindow.Bottom - csbi.srWindow.Top + 1)
return 0
proc terminalWidth*(): int =
var w: int = 0
w = terminalWidthIoctl([ getStdHandle(STD_INPUT_HANDLE),
@@ -91,6 +98,14 @@ when defined(windows):
if w > 0: return w
return 80
proc terminalHeight*(): int =
var h: int = 0
h = terminalHeightIoctl([ getStdHandle(STD_INPUT_HANDLE),
getStdHandle(STD_OUTPUT_HANDLE),
getStdHandle(STD_ERROR_HANDLE) ] )
if h > 0: return h
return 0
proc setConsoleCursorPosition(hConsoleOutput: HANDLE,
dwCursorPosition: COORD): WINBOOL{.
stdcall, dynlib: "kernel32", importc: "SetConsoleCursorPosition".}
@@ -177,7 +192,17 @@ else:
return int(win.ws_col)
return 0
proc terminalHeightIoctl*(fds: openArray[int]): int =
## Returns terminal height from first fd that supports the ioctl.
var win: IOctl_WinSize
for fd in fds:
if ioctl(cint(fd), TIOCGWINSZ, addr win) != -1:
return int(win.ws_row)
return 0
var L_ctermid{.importc, header: "<stdio.h>".}: cint
proc terminalWidth*(): int =
## Returns some reasonable terminal width from either standard file
## descriptors, controlling terminal, environment variables or tradition.
@@ -195,6 +220,29 @@ else:
return w
return 80 #Finally default to venerable value
proc terminalHeight*(): int =
## Returns some reasonable terminal height from either standard file
## descriptors, controlling terminal, environment variables or tradition.
## Zero is returned if the height could not be determined.
var h = terminalHeightIoctl([0, 1, 2]) # Try standard file descriptors
if h > 0: return h
var cterm = newString(L_ctermid) # Try controlling tty
var fd = open(ctermid(cstring(cterm)), O_RDONLY)
if fd != -1:
h = terminalHeightIoctl([ int(fd) ])
discard close(fd)
if h > 0: return h
var s = getEnv("LINES") # Try standard env var
if len(s) > 0 and parseInt(string(s), h) > 0 and h > 0:
return h
return 0 # Could not determine height
proc terminalSize*(): tuple[w, h: int] =
## Returns the terminal width and height as a tuple. Internally calls
## `terminalWidth` and `terminalHeight`, so the same assumptions apply.
result = (terminalWidth(), terminalHeight())
when defined(windows):
proc setCursorVisibility(f: File, visible: bool) =
var ccsi: CONSOLE_CURSOR_INFO

View File

@@ -91,13 +91,24 @@ proc rawTag*(n: XmlNode): string {.inline.} =
shallowCopy(result, n.fTag)
proc innerText*(n: XmlNode): string =
## gets the inner text of `n`. `n` has to be an ``xnElement`` node. Only
## ``xnText`` and ``xnEntity`` nodes are considered part of `n`'s inner text,
## other child nodes are silently ignored.
## gets the inner text of `n`:
##
## - If `n` is `xnText` or `xnEntity`, returns its content.
## - If `n` is `xnElement`, runs recursively on each child node and
## concatenates the results.
## - Otherwise returns an empty string.
proc worker(res: var string, n: XmlNode) =
case n.k
of xnText, xnEntity:
res.add(n.fText)
of xnElement:
for sub in n.s:
worker(res, sub)
else:
discard
result = ""
assert n.k == xnElement
for i in 0 .. n.s.len-1:
if n.s[i].k in {xnText, xnEntity}: result.add(n.s[i].fText)
worker(result, n)
proc tag*(n: XmlNode): string {.inline.} =
## gets the tag name of `n`. `n` has to be an ``xnElement`` node.

View File

@@ -153,13 +153,19 @@ template setColor(c, col) =
proc writeCell(msg: cstring, c: PCell) =
var kind = -1
if c.typ != nil: kind = ord(c.typ.kind)
var typName: cstring = "nil"
if c.typ != nil:
kind = ord(c.typ.kind)
when defined(nimTypeNames):
if not c.typ.name.isNil:
typName = c.typ.name
when leakDetector:
c_fprintf(stdout, "[GC] %s: %p %d rc=%ld from %s(%ld)\n",
msg, c, kind, c.refcount shr rcShift, c.filename, c.line)
c_fprintf(stdout, "[GC] %s: %p %d %s rc=%ld from %s(%ld)\n",
msg, c, kind, typName, c.refcount shr rcShift, c.filename, c.line)
else:
c_fprintf(stdout, "[GC] %s: %p %d rc=%ld; color=%ld\n",
msg, c, kind, c.refcount shr rcShift, c.color)
c_fprintf(stdout, "[GC] %s: %p %d %s rc=%ld; color=%ld\n",
msg, c, kind, typName, c.refcount shr rcShift, c.color)
template gcTrace(cell, state: expr): stmt {.immediate.} =
when traceGC: traceCell(cell, state)

View File

@@ -293,6 +293,8 @@ proc SSL_get_error*(s: SslPtr, ret_code: cInt): cInt{.cdecl, dynlib: DLLSSLName,
proc SSL_accept*(ssl: SslPtr): cInt{.cdecl, dynlib: DLLSSLName, importc.}
proc SSL_pending*(ssl: SslPtr): cInt{.cdecl, dynlib: DLLSSLName, importc.}
proc BIO_new_mem_buf*(data: pointer, len: cint): BIO{.cdecl,
dynlib: DLLSSLName, importc.}
proc BIO_new_ssl_connect*(ctx: SslCtx): BIO{.cdecl,
dynlib: DLLSSLName, importc.}
proc BIO_ctrl*(bio: BIO, cmd: cint, larg: int, arg: cstring): int{.cdecl,

View File

@@ -0,0 +1,28 @@
import unittest
import httpcore
suite "httpcore":
test "HttpCode":
assert $Http418 == "418 I'm a teapot"
assert Http418.is4xx() == true
assert Http418.is2xx() == false
test "headers":
var h = newHttpHeaders()
assert h.len == 0
h.add("Cookie", "foo")
assert h.len == 1
assert h.hasKey("cooKIE")
assert h["Cookie"] == "foo"
assert h["cookie"] == "foo"
h["cookie"] = @["bar", "x"]
assert h["Cookie"] == "bar"
assert h["Cookie", 1] == "x"
assert h["Cookie"].contains("BaR") == true
assert h["Cookie"].contains("X") == true
assert "baR" in h["cookiE"]
h.del("coOKie")
assert h.len == 0

View File

@@ -1,6 +1,11 @@
discard """
file: "txmltree.nim"
output: "true"
output: '''true
true
true
true
true
'''
"""
import xmltree, strtabs
@@ -9,5 +14,14 @@ var x = <>a(href="nim.de", newText("www.nim-test.de"))
echo($x == "<a href=\"nim.de\">www.nim-test.de</a>")
echo(newText("foo").innerText == "foo")
echo(newEntity("bar").innerText == "bar")
echo(newComment("baz").innerText == "")
let y = newXmlTree("x", [
newText("foo"),
newXmlTree("y", [
newText("bar")
])
])
echo(y.innerText == "foobar")

View File

@@ -39,7 +39,8 @@ Library Additions
``deques`` provides a superset of ``queues`` API with clear naming.
``queues`` module is now deprecated and will be removed in the future.
- Added ``hideCursor`` and ``showCursor`` to the ``terminal``
- Added ``hideCursor``, ``showCursor``, ``terminalWidth``,
``terminalWidthIoctl`` and ``terminalSize`` to the ``terminal``
`(doc) <http://nim-lang.org/docs/terminal.html>`_ module.