123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178 |
- #
- #
- # Nim's Runtime Library
- # (c) Copyright 2010 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- ## This module implements a base64 encoder and decoder.
- ##
- ## Encoding data
- ## -------------
- ##
- ## In order to encode some text simply call the ``encode`` procedure:
- ##
- ## .. code-block::nim
- ## import base64
- ## let encoded = encode("Hello World")
- ## echo(encoded) # SGVsbG8gV29ybGQ=
- ##
- ## Apart from strings you can also encode lists of integers or characters:
- ##
- ## .. code-block::nim
- ## import base64
- ## let encodedInts = encode([1,2,3])
- ## echo(encodedInts) # AQID
- ## let encodedChars = encode(['h','e','y'])
- ## echo(encodedChars) # aGV5
- ##
- ## The ``encode`` procedure takes an ``openarray`` so both arrays and sequences
- ## can be passed as parameters.
- ##
- ## Decoding data
- ## -------------
- ##
- ## To decode a base64 encoded data string simply call the ``decode``
- ## procedure:
- ##
- ## .. code-block::nim
- ## import base64
- ## echo(decode("SGVsbG8gV29ybGQ=")) # Hello World
- const
- cb64 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"
- template encodeInternal(s: typed, lineLen: int, newLine: string): untyped =
- ## encodes `s` into base64 representation. After `lineLen` characters, a
- ## `newline` is added.
- var total = ((len(s) + 2) div 3) * 4
- let numLines = (total + lineLen - 1) div lineLen
- if numLines > 0: inc(total, (numLines - 1) * newLine.len)
- result = newString(total)
- var
- i = 0
- r = 0
- currLine = 0
- while i < s.len - 2:
- let
- a = ord(s[i])
- b = ord(s[i+1])
- c = ord(s[i+2])
- result[r] = cb64[a shr 2]
- result[r+1] = cb64[((a and 3) shl 4) or ((b and 0xF0) shr 4)]
- result[r+2] = cb64[((b and 0x0F) shl 2) or ((c and 0xC0) shr 6)]
- result[r+3] = cb64[c and 0x3F]
- inc(r, 4)
- inc(i, 3)
- inc(currLine, 4)
- # avoid index out of bounds when lineLen == encoded length
- if currLine >= lineLen and i != s.len-2 and r < total:
- for x in items(newLine):
- result[r] = x
- inc(r)
- currLine = 0
- if i < s.len-1:
- let
- a = ord(s[i])
- b = ord(s[i+1])
- result[r] = cb64[a shr 2]
- result[r+1] = cb64[((a and 3) shl 4) or ((b and 0xF0) shr 4)]
- result[r+2] = cb64[((b and 0x0F) shl 2)]
- result[r+3] = '='
- if r+4 != result.len:
- setLen(result, r+4)
- elif i < s.len:
- let a = ord(s[i])
- result[r] = cb64[a shr 2]
- result[r+1] = cb64[(a and 3) shl 4]
- result[r+2] = '='
- result[r+3] = '='
- if r+4 != result.len:
- setLen(result, r+4)
- else:
- if r != result.len:
- setLen(result, r)
- #assert(r == result.len)
- discard
- proc encode*[T:SomeInteger|char](s: openarray[T], lineLen = 75, newLine="\13\10"): string =
- ## encodes `s` into base64 representation. After `lineLen` characters, a
- ## `newline` is added.
- ##
- ## This procedure encodes an openarray (array or sequence) of either integers
- ## or characters.
- encodeInternal(s, lineLen, newLine)
- proc encode*(s: string, lineLen = 75, newLine="\13\10"): string =
- ## encodes `s` into base64 representation. After `lineLen` characters, a
- ## `newline` is added.
- ##
- ## This procedure encodes a string.
- encodeInternal(s, lineLen, newLine)
- proc decodeByte(b: char): int {.inline.} =
- case b
- of '+': result = ord('>')
- of '0'..'9': result = ord(b) + 4
- of 'A'..'Z': result = ord(b) - ord('A')
- of 'a'..'z': result = ord(b) - 71
- else: result = 63
- proc decode*(s: string): string =
- ## decodes a string in base64 representation back into its original form.
- ## Whitespace is skipped.
- const Whitespace = {' ', '\t', '\v', '\r', '\l', '\f'}
- var total = ((len(s) + 3) div 4) * 3
- # total is an upper bound, as we will skip arbitrary whitespace:
- result = newString(total)
- var
- i = 0
- r = 0
- while true:
- while i < s.len and s[i] in Whitespace: inc(i)
- if i < s.len-3:
- let
- a = s[i].decodeByte
- b = s[i+1].decodeByte
- c = s[i+2].decodeByte
- d = s[i+3].decodeByte
- result[r] = chr((a shl 2) and 0xff or ((b shr 4) and 0x03))
- result[r+1] = chr((b shl 4) and 0xff or ((c shr 2) and 0x0F))
- result[r+2] = chr((c shl 6) and 0xff or (d and 0x3F))
- inc(r, 3)
- inc(i, 4)
- else: break
- assert i == s.len
- # adjust the length:
- if i > 0 and s[i-1] == '=':
- dec(r)
- if i > 1 and s[i-2] == '=': dec(r)
- setLen(result, r)
- when isMainModule:
- assert encode("leasure.") == "bGVhc3VyZS4="
- assert encode("easure.") == "ZWFzdXJlLg=="
- assert encode("asure.") == "YXN1cmUu"
- assert encode("sure.") == "c3VyZS4="
- const testInputExpandsTo76 = "+++++++++++++++++++++++++++++++++++++++++++++++++++++++++"
- const testInputExpands = "++++++++++++++++++++++++++++++"
- const longText = """Man is distinguished, not only by his reason, but by this
- singular passion from other animals, which is a lust of the mind,
- that by a perseverance of delight in the continued and indefatigable
- generation of knowledge, exceeds the short vehemence of any carnal
- pleasure."""
- const tests = ["", "abc", "xyz", "man", "leasure.", "sure.", "easure.",
- "asure.", longText, testInputExpandsTo76, testInputExpands]
- for t in items(tests):
- assert decode(encode(t)) == t
- assert decode(encode(t, lineLen=40)) == t
- assert decode(encode(t, lineLen=76)) == t
|