123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- #
- #
- # 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.
- ##
- ## Unstable API.
- ##
- ## Base64 is an encoding and decoding technique used to convert binary
- ## data to an ASCII string format.
- ## Each Base64 digit represents exactly 6 bits of data. Three 8-bit
- ## bytes (i.e., a total of 24 bits) can therefore be represented by
- ## four 6-bit Base64 digits.
- ##
- ## Basic usage
- ## ===========
- ##
- ## Encoding data
- ## -------------
- ##
- ## .. code-block::nim
- ## import base64
- ## let encoded = encode("Hello World")
- ## assert 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])
- ## assert encodedInts == "AQID"
- ## let encodedChars = encode(['h','e','y'])
- ## assert encodedChars == "aGV5"
- ##
- ##
- ## Decoding data
- ## -------------
- ##
- ## .. code-block::nim
- ## import base64
- ## let decoded = decode("SGVsbG8gV29ybGQ=")
- ## assert decoded == "Hello World"
- ##
- ## URL Safe Base64
- ## ---------------
- ##
- ## .. code-block::nim
- ## import base64
- ## doAssert encode("c\xf7>", safe = true) == "Y_c-"
- ## doAssert encode("c\xf7>", safe = false) == "Y/c+"
- ##
- ## See also
- ## ========
- ##
- ## * `hashes module<hashes.html>`_ for efficient computations of hash values for diverse Nim types
- ## * `md5 module<md5.html>`_ implements the MD5 checksum algorithm
- ## * `sha1 module<sha1.html>`_ implements a sha1 encoder and decoder
- template cbBase(a, b): untyped = [
- 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M',
- 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
- 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm',
- 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z',
- '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', a, b]
- let
- cb64 = cbBase('+', '/')
- cb64safe = cbBase('-', '_')
- const
- cb64VM = cbBase('+', '/')
- cb64safeVM = cbBase('-', '_')
- const
- invalidChar = 255
- template encodeSize(size: int): int = (size * 4 div 3) + 6
- template encodeInternal(s, alphabet: typed): untyped =
- ## encodes `s` into base64 representation.
- result.setLen(encodeSize(s.len))
- var
- inputIndex = 0
- outputIndex = 0
- inputEnds = s.len - s.len mod 3
- n: uint32
- b: uint32
- template inputByte(exp: untyped) =
- b = uint32(s[inputIndex])
- n = exp
- inc inputIndex
- template outputChar(x: typed) =
- result[outputIndex] = alphabet[x and 63]
- inc outputIndex
- template outputChar(c: char) =
- result[outputIndex] = c
- inc outputIndex
- while inputIndex != inputEnds:
- inputByte(b shl 16)
- inputByte(n or b shl 8)
- inputByte(n or b shl 0)
- outputChar(n shr 18)
- outputChar(n shr 12)
- outputChar(n shr 6)
- outputChar(n shr 0)
- var padding = s.len mod 3
- if padding == 1:
- inputByte(b shl 16)
- outputChar(n shr 18)
- outputChar(n shr 12)
- outputChar('=')
- outputChar('=')
- elif padding == 2:
- inputByte(b shl 16)
- inputByte(n or b shl 8)
- outputChar(n shr 18)
- outputChar(n shr 12)
- outputChar(n shr 6)
- outputChar('=')
- result.setLen(outputIndex)
- template encodeImpl() {.dirty.} =
- when nimVM:
- block:
- let lookupTableVM = if safe: cb64safeVM else: cb64VM
- encodeInternal(s, lookupTableVM)
- else:
- block:
- let lookupTable = if safe: unsafeAddr(cb64safe) else: unsafeAddr(cb64)
- encodeInternal(s, lookupTable)
- proc encode*[T: SomeInteger|char](s: openArray[T], safe = false): string =
- ## Encodes `s` into base64 representation.
- ##
- ## This procedure encodes an openarray (array or sequence) of either integers
- ## or characters.
- ##
- ## If ``safe`` is ``true`` then it will encode using the
- ## URL-Safe and Filesystem-safe standard alphabet characters,
- ## which substitutes ``-`` instead of ``+`` and ``_`` instead of ``/``.
- ## * https://en.wikipedia.org/wiki/Base64#URL_applications
- ## * https://tools.ietf.org/html/rfc4648#page-7
- ##
- ## **See also:**
- ## * `encode proc<#encode,string>`_ for encoding a string
- ## * `decode proc<#decode,string>`_ for decoding a string
- runnableExamples:
- assert encode(['n', 'i', 'm']) == "bmlt"
- assert encode(@['n', 'i', 'm']) == "bmlt"
- assert encode([1, 2, 3, 4, 5]) == "AQIDBAU="
- encodeImpl()
- proc encode*(s: string, safe = false): string =
- ## Encodes ``s`` into base64 representation.
- ##
- ## This procedure encodes a string.
- ##
- ## If ``safe`` is ``true`` then it will encode using the
- ## URL-Safe and Filesystem-safe standard alphabet characters,
- ## which substitutes ``-`` instead of ``+`` and ``_`` instead of ``/``.
- ## * https://en.wikipedia.org/wiki/Base64#URL_applications
- ## * https://tools.ietf.org/html/rfc4648#page-7
- ##
- ## **See also:**
- ## * `encode proc<#encode,openArray[T]>`_ for encoding an openarray
- ## * `decode proc<#decode,string>`_ for decoding a string
- runnableExamples:
- assert encode("Hello World") == "SGVsbG8gV29ybGQ="
- encodeImpl()
- proc encodeMime*(s: string, lineLen = 75, newLine = "\r\n"): string =
- ## Encodes ``s`` into base64 representation as lines.
- ## Used in email MIME format, use ``lineLen`` and ``newline``.
- ##
- ## This procedure encodes a string according to MIME spec.
- ##
- ## **See also:**
- ## * `encode proc<#encode,string>`_ for encoding a string
- ## * `decode proc<#decode,string>`_ for decoding a string
- runnableExamples:
- assert encodeMime("Hello World", 4, "\n") == "SGVs\nbG8g\nV29y\nbGQ="
- for i, c in encode(s):
- if i != 0 and (i mod lineLen == 0):
- result.add(newLine)
- result.add(c)
- proc initDecodeTable*(): array[256, char] =
- # computes a decode table at compile time
- for i in 0 ..< 256:
- let ch = char(i)
- var code = invalidChar
- if ch >= 'A' and ch <= 'Z': code = i - 0x00000041
- if ch >= 'a' and ch <= 'z': code = i - 0x00000047
- if ch >= '0' and ch <= '9': code = i + 0x00000004
- if ch == '+' or ch == '-': code = 0x0000003E
- if ch == '/' or ch == '_': code = 0x0000003F
- result[i] = char(code)
- const
- decodeTable = initDecodeTable()
- proc decode*(s: string): string =
- ## Decodes string ``s`` in base64 representation back into its original form.
- ## The initial whitespace is skipped.
- ##
- ## **See also:**
- ## * `encode proc<#encode,openArray[T]>`_ for encoding an openarray
- ## * `encode proc<#encode,string>`_ for encoding a string
- runnableExamples:
- assert decode("SGVsbG8gV29ybGQ=") == "Hello World"
- assert decode(" SGVsbG8gV29ybGQ=") == "Hello World"
- if s.len == 0: return
- proc decodeSize(size: int): int =
- return (size * 3 div 4) + 6
- template inputChar(x: untyped) =
- let x = int decodeTable[ord(s[inputIndex])]
- if x == invalidChar:
- raise newException(ValueError,
- "Invalid base64 format character `" & s[inputIndex] &
- "` (ord " & $s[inputIndex].ord & ") at location " & $inputIndex & ".")
- inc inputIndex
- template outputChar(x: untyped) =
- result[outputIndex] = char(x and 255)
- inc outputIndex
- # pre allocate output string once
- result.setLen(decodeSize(s.len))
- var
- inputIndex = 0
- outputIndex = 0
- inputLen = s.len
- inputEnds = 0
- # strip trailing characters
- while s[inputLen - 1] in {'\n', '\r', ' ', '='}:
- dec inputLen
- # hot loop: read 4 characters at at time
- inputEnds = inputLen - 4
- while inputIndex <= inputEnds:
- while s[inputIndex] in {'\n', '\r', ' '}:
- inc inputIndex
- inputChar(a)
- inputChar(b)
- inputChar(c)
- inputChar(d)
- outputChar(a shl 2 or b shr 4)
- outputChar(b shl 4 or c shr 2)
- outputChar(c shl 6 or d shr 0)
- # do the last 2 or 3 characters
- var leftLen = abs((inputIndex - inputLen) mod 4)
- if leftLen == 2:
- inputChar(a)
- inputChar(b)
- outputChar(a shl 2 or b shr 4)
- elif leftLen == 3:
- inputChar(a)
- inputChar(b)
- inputChar(c)
- outputChar(a shl 2 or b shr 4)
- outputChar(b shl 4 or c shr 2)
- result.setLen(outputIndex)
|