diff options
author | Simon Hafner <hafnersimon@gmail.com> | 2011-11-03 17:05:32 +0100 |
---|---|---|
committer | Simon Hafner <hafnersimon@gmail.com> | 2011-12-22 17:17:06 +0100 |
commit | 5ede29cd051b818c7c2fc926c4acba86ddb76a0e (patch) | |
tree | 8e0f21121ffd9f44884e2354a63ede3313c0e05b | |
parent | 4f08946f64f04ef85a53c686fb89b8d84fbc6d65 (diff) | |
download | Nim-5ede29cd051b818c7c2fc926c4acba86ddb76a0e.tar.gz |
implemented readAll()
-rwxr-xr-x | lib/system.nim | 9 | ||||
-rwxr-xr-x | lib/system/sysio.nim | 64 | ||||
-rw-r--r-- | tests/system/helpers/readall_echo.nim | 1 | ||||
-rw-r--r-- | tests/system/io.nim | 25 |
4 files changed, 77 insertions, 22 deletions
diff --git a/lib/system.nim b/lib/system.nim index 8a99781cc..41100639f 100755 --- a/lib/system.nim +++ b/lib/system.nim @@ -1675,10 +1675,13 @@ when not defined(EcmaScript) and not defined(NimrodVM): proc FlushFile*(f: TFile) {.importc: "fflush", noDecl.} ## Flushes `f`'s buffer. + proc readAll*(file: TFile): TaintedString + ## Reads all data from the stream `file`. Raises an IO exception + ## in case of an Error + proc readFile*(filename: string): TaintedString - ## Opens a file named `filename` for reading. Then reads the - ## file's content completely into a string and - ## closes the file afterwards. Returns the string. + ## Opens a file named `filename` for reading. Then calls readAll + ## and closes the file afterwards. Returns the string. ## Raises an IO exception in case of an error. proc writeFile*(filename, content: string) diff --git a/lib/system/sysio.nim b/lib/system/sysio.nim index 890f526ea..6641d7218 100755 --- a/lib/system/sysio.nim +++ b/lib/system/sysio.nim @@ -24,6 +24,13 @@ proc putc(c: Char, stream: TFile) {.importc: "putc", nodecl.} proc fprintf(f: TFile, frmt: CString) {.importc: "fprintf", nodecl, varargs.} proc strlen(c: cstring): int {.importc: "strlen", nodecl.} + +# C routine that is used here: +proc fread(buf: Pointer, size, n: int, f: TFile): int {. + importc: "fread", noDecl.} +proc fseek(f: TFile, offset: clong, whence: int): int {. + importc: "fseek", noDecl.} +proc ftell(f: TFile): int {.importc: "ftell", noDecl.} proc setvbuf(stream: TFile, buf: pointer, typ, size: cint): cint {. importc, nodecl.} @@ -36,6 +43,9 @@ var IOFBF {.importc: "_IOFBF", nodecl.}: cint IONBF {.importc: "_IONBF", nodecl.}: cint +const + BUF_SIZE = 4000 + proc raiseEIO(msg: string) {.noinline, noreturn.} = raise newException(EIO, msg) @@ -82,21 +92,44 @@ proc write(f: TFile, c: Char) = putc(c, f) proc write(f: TFile, a: openArray[string]) = for x in items(a): write(f, x) +proc readAllBuffer(file: TFile): string = + # This proc is for TFile we want to read but don't know how many + # bytes we need to read before the buffer is empty. + result = "" + var buffer = newString(BUF_SIZE) + var bytesRead = BUF_SIZE + while bytesRead == BUF_SIZE: + bytesRead = readBuffer(file, addr(buffer[0]), BUF_SIZE) + result.add(buffer) + +proc readAllFile(file: TFile): string = + # We aquire the filesize beforehand and hope it doesn't change. + # Speeds things up. + var len = getFileSize(file) + if len >= high(int): + raiseEIO("file too big to fit in memory") + result = newString(int(len)) + if readBuffer(file, addr(result[0]), int(len)) != len: + raiseEIO("error while reading from file") + +proc hasDefinedLength(file: TFile): bool = + var oldPos = getFilePos(file) + discard fseek(file, 0, 2) # seek the end of the file + result = ftell(file) >= 0 + setFilePos(file, oldPos) + +proc readAll(file: TFile): TaintedString = + # Separate handling needed because we need to buffer when we + # don't know the overall length of the TFile. + if hasDefinedLength(file): + result = readAllBuffer(file).TaintedSTring + else: + result = readAllFile(file).TaintedString + proc readFile(filename: string): TaintedString = var f = open(filename) try: - var len = getFileSize(f) - if len < high(int): - when taintMode: - result = newString(int(len)).TaintedString - if readBuffer(f, addr(string(result)[0]), int(len)) != len: - raiseEIO("error while reading from file") - else: - result = newString(int(len)) - if readBuffer(f, addr(result[0]), int(len)) != len: - raiseEIO("error while reading from file") - else: - raiseEIO("file too big to fit in memory") + result = readAllFile(f) finally: close(f) @@ -157,13 +190,6 @@ proc open(f: var TFile, filehandle: TFileHandle, mode: TFileMode): bool = f = fdopen(filehandle, FormatOpen[mode]) result = f != nil -# C routine that is used here: -proc fread(buf: Pointer, size, n: int, f: TFile): int {. - importc: "fread", noDecl.} -proc fseek(f: TFile, offset: clong, whence: int): int {. - importc: "fseek", noDecl.} -proc ftell(f: TFile): int {.importc: "ftell", noDecl.} - proc fwrite(buf: Pointer, size, n: int, f: TFile): int {. importc: "fwrite", noDecl.} diff --git a/tests/system/helpers/readall_echo.nim b/tests/system/helpers/readall_echo.nim new file mode 100644 index 000000000..2890217ef --- /dev/null +++ b/tests/system/helpers/readall_echo.nim @@ -0,0 +1 @@ +echo(stdin.readAll) diff --git a/tests/system/io.nim b/tests/system/io.nim new file mode 100644 index 000000000..1c3be47f4 --- /dev/null +++ b/tests/system/io.nim @@ -0,0 +1,25 @@ +import + unittest, osproc, streams, os +const STRING_DATA = "Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet. Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet." +const TEST_FILE = "tests/testdata/string" + +proc echoLoop(str: string): string = + result = "" + var process = startProcess("tests/system/helpers/readall_echo") + var input = process.inputStream + input.write(str) + input.close() + var output = process.outputStream + discard process.waitForExit + while not output.atEnd: + result.add(output.readLine) + +suite "io": + suite "readAll": + test "stdin": + check: + echoLoop(STRING_DATA) == STRING_DATA + echoLoop(STRING_DATA[0..3999]) == STRING_DATA[0..3999] + test "file": + check: + readFile(TEST_FILE) == STRING_DATA |