1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174
|
# base64.tcl --
#
# Encode/Decode base64 for a string
# Stephen Uhler / Brent Welch (c) 1997 Sun Microsystems
# The decoder was done for exmh by Chris Garrigues
#
# Copyright (c) 1998-2000 by Ajuba Solutions.
# See the file "license.terms" for information on usage and redistribution
# of this file, and for a DISCLAIMER OF ALL WARRANTIES.
#
# RCS: @(#) $Id: base64.tcl,v 1.2 2001/08/29 18:46:43 blezek Exp $
# Version 1.0 implemented Base64_Encode, Bae64_Decode
# Version 2.0 uses the base64 namespace
# Version 2.1 fixes various decode bugs and adds options to encode
namespace eval base64 {
variable i 0
variable char
variable base64
variable base64_en
foreach char {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 + /} {
set base64($char) $i
set base64_en($i) $char
incr i
}
namespace export *
}
# base64::encode --
#
# Base64 encode a given string.
#
# Arguments:
# args ?-maxlen maxlen? ?-wrapchar wrapchar? string
#
# If maxlen is 0, the output is not wrapped.
#
# Results:
# A Base64 encoded version of $string, wrapped at $maxlen characters
# by $wrapchar.
proc base64::encode {args} {
variable base64_en
# Set the default wrapchar and maximum line length to match the output
# of GNU uuencode 4.2. Various RFC's allow for different wrapping
# characters and wraplengths, so these may be overridden by command line
# options.
set wrapchar "\n"
set maxlen 60
if { [llength $args] == 0 } {
error "wrong # args: should be \"[lindex [info level 0] 0]\
?-maxlen maxlen? ?-wrapchar wrapchar? string\""
}
set optionStrings [list "-maxlen" "-wrapchar"]
for {set i 0} {$i < [llength $args] - 1} {incr i} {
set arg [lindex $args $i]
set index [lsearch -glob $optionStrings "${arg}*"]
if { $index == -1 } {
error "unknown option \"$arg\": must be -maxlen or -wrapchar"
}
incr i
if { $i >= [llength $args] - 1 } {
error "value for \"$arg\" missing"
}
set val [lindex $args $i]
set [string range [lindex $optionStrings $index] 1 end] $val
}
if { ![string is integer -strict $maxlen] } {
error "expected integer but got \"$maxlen\""
}
set string [lindex $args end]
set result {}
set state 0
set length 0
foreach {c} [split $string {}] {
# Do the line length check before appending so that we don't get an
# extra newline if the output is a multiple of $maxlen chars long.
if {$maxlen && $length >= $maxlen} {
append result $wrapchar
set length 0
}
scan $c %c x
switch [incr state] {
1 { append result $base64_en([expr {($x >>2) & 0x3F}]) }
2 { append result \
$base64_en([expr {(($old << 4) & 0x30) | (($x >> 4) & 0xF)}]) }
3 { append result \
$base64_en([expr {(($old << 2) & 0x3C) | (($x >> 6) & 0x3)}])
append result $base64_en([expr {($x & 0x3F)}])
incr length
set state 0}
}
set old $x
incr length
}
set x 0
switch $state {
0 { # OK }
1 { append result $base64_en([expr {(($old << 4) & 0x30)}])== }
2 { append result $base64_en([expr {(($old << 2) & 0x3C)}])= }
}
return $result
}
# base64::decode --
#
# Base64 decode a given string.
#
# Arguments:
# string The string to decode. Characters not in the base64
# alphabet are ignored (e.g., newlines)
#
# Results:
# The decoded value.
proc base64::decode {string} {
variable base64
set output {}
set group 0
set j 18
foreach char [split $string {}] {
if {[string compare $char "="]} {
# RFC 2045 says that line breaks and other characters not part
# of the Base64 alphabet must be ignored, and that the decoder
# can optionally emit a warning or reject the message. We opt
# not to do so, but to just ignore the character.
if { ![info exists base64($char)] } {
continue
}
set bits $base64($char)
set group [expr {$group | ($bits << $j)}]
if {[incr j -6] < 0} {
scan [format %06x $group] %2x%2x%2x a b c
append output [format %c%c%c $a $b $c]
set group 0
set j 18
}
} else {
# = indicates end of data. Output whatever chars are left.
# The encoding algorithm dictates that we can only have 1 or 2
# padding characters. If j is 6, we have 12 bits of input
# (enough for 1 8-bit output). If j is 0, we have 18 bits of
# input (enough for 2 8-bit outputs).
# It is crucial to scan three hex digits even though we
# discard c - older code used %04x and scanned 2 hex digits
# but really ended up generating 5 or 6 (not 4!) digits and
# resulted in alignment errors.
scan [format %06x $group] %2x%2x%2x a b c
if {$j == 6} {
append output [format %c $a]
} elseif {$j == 0} {
append output [format %c%c $a $b]
}
break
}
}
return $output
}
package provide base64 2.1
|