Base64 works with streams, ignores newlines in inputs and can output newlines when appropriate
							parent
							
								
									5909ca0bd8
								
							
						
					
					
						commit
						3b679cf2be
					
				|  | @ -1,4 +1,4 @@ | ||||||
| USING: kernel tools.test base64 strings ; | USING: kernel tools.test base64 strings sequences  ; | ||||||
| IN: base64.tests | IN: base64.tests | ||||||
| 
 | 
 | ||||||
| [ "abcdefghijklmnopqrstuvwxyz" ] [ "abcdefghijklmnopqrstuvwxyz" >base64 base64> >string | [ "abcdefghijklmnopqrstuvwxyz" ] [ "abcdefghijklmnopqrstuvwxyz" >base64 base64> >string | ||||||
|  | @ -7,6 +7,7 @@ IN: base64.tests | ||||||
| [ "a" ] [ "a" >base64 base64> >string ] unit-test | [ "a" ] [ "a" >base64 base64> >string ] unit-test | ||||||
| [ "ab" ] [ "ab" >base64 base64> >string ] unit-test | [ "ab" ] [ "ab" >base64 base64> >string ] unit-test | ||||||
| [ "abc" ] [ "abc" >base64 base64> >string ] unit-test | [ "abc" ] [ "abc" >base64 base64> >string ] unit-test | ||||||
|  | [ "abcde" ] [ "abcde" >base64 3 cut "\r\n" swap 3append base64> >string ] unit-test | ||||||
| 
 | 
 | ||||||
| ! From http://en.wikipedia.org/wiki/Base64 | ! From http://en.wikipedia.org/wiki/Base64 | ||||||
| [ "TWFuIGlzIGRpc3Rpbmd1aXNoZWQsIG5vdCBvbmx5IGJ5IGhpcyByZWFzb24sIGJ1dCBieSB0aGlzIHNpbmd1bGFyIHBhc3Npb24gZnJvbSBvdGhlciBhbmltYWxzLCB3aGljaCBpcyBhIGx1c3Qgb2YgdGhlIG1pbmQsIHRoYXQgYnkgYSBwZXJzZXZlcmFuY2Ugb2YgZGVsaWdodCBpbiB0aGUgY29udGludWVkIGFuZCBpbmRlZmF0aWdhYmxlIGdlbmVyYXRpb24gb2Yga25vd2xlZGdlLCBleGNlZWRzIHRoZSBzaG9ydCB2ZWhlbWVuY2Ugb2YgYW55IGNhcm5hbCBwbGVhc3VyZS4=" ] | [ "TWFuIGlzIGRpc3Rpbmd1aXNoZWQsIG5vdCBvbmx5IGJ5IGhpcyByZWFzb24sIGJ1dCBieSB0aGlzIHNpbmd1bGFyIHBhc3Npb24gZnJvbSBvdGhlciBhbmltYWxzLCB3aGljaCBpcyBhIGx1c3Qgb2YgdGhlIG1pbmQsIHRoYXQgYnkgYSBwZXJzZXZlcmFuY2Ugb2YgZGVsaWdodCBpbiB0aGUgY29udGludWVkIGFuZCBpbmRlZmF0aWdhYmxlIGdlbmVyYXRpb24gb2Yga25vd2xlZGdlLCBleGNlZWRzIHRoZSBzaG9ydCB2ZWhlbWVuY2Ugb2YgYW55IGNhcm5hbCBwbGVhc3VyZS4=" ] | ||||||
|  | @ -15,5 +16,11 @@ IN: base64.tests | ||||||
|     >base64 >string |     >base64 >string | ||||||
| ] unit-test | ] unit-test | ||||||
| 
 | 
 | ||||||
|  | [ "TWFuIGlzIGRpc3Rpbmd1aXNoZWQsIG5vdCBvbmx5IGJ5IGhpcyByZWFzb24sIGJ1dCBieSB0aGlz\r\nIHNpbmd1bGFyIHBhc3Npb24gZnJvbSBvdGhlciBhbmltYWxzLCB3aGljaCBpcyBhIGx1c3Qgb2Yg\r\ndGhlIG1pbmQsIHRoYXQgYnkgYSBwZXJzZXZlcmFuY2Ugb2YgZGVsaWdodCBpbiB0aGUgY29udGlu\r\ndWVkIGFuZCBpbmRlZmF0aWdhYmxlIGdlbmVyYXRpb24gb2Yga25vd2xlZGdlLCBleGNlZWRzIHRo\r\nZSBzaG9ydCB2ZWhlbWVuY2Ugb2YgYW55IGNhcm5hbCBwbGVhc3VyZS4=" ] | ||||||
|  | [ | ||||||
|  |     "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." | ||||||
|  |     >base64-lines >string | ||||||
|  | ] unit-test | ||||||
|  | 
 | ||||||
| \ >base64 must-infer | \ >base64 must-infer | ||||||
| \ base64> must-infer | \ base64> must-infer | ||||||
|  |  | ||||||
|  | @ -1,16 +1,22 @@ | ||||||
| ! Copyright (C) 2008 Doug Coleman. | ! Copyright (C) 2008 Doug Coleman, Daniel Ehrenberg. | ||||||
| ! See http://factorcode.org/license.txt for BSD license. | ! See http://factorcode.org/license.txt for BSD license. | ||||||
| USING: kernel math sequences io.binary splitting grouping | USING: combinators io io.binary io.encodings.binary | ||||||
| accessors ; | io.streams.byte-array io.streams.string kernel math namespaces | ||||||
|  | sequences strings ; | ||||||
| IN: base64 | IN: base64 | ||||||
| 
 | 
 | ||||||
| <PRIVATE | <PRIVATE | ||||||
| 
 | 
 | ||||||
| : count-end ( seq quot -- n ) | : read1-ignoring ( ignoring -- ch ) | ||||||
|     trim-right-slice [ seq>> length ] [ to>> ] bi - ; inline |     read1 2dup swap member? [ drop read1-ignoring ] [ nip ] if ; | ||||||
|  | 
 | ||||||
|  | : read-ignoring ( ignoring n -- str ) | ||||||
|  |     [ drop read1-ignoring ] with map harvest | ||||||
|  |     [ f ] [ >string ] if-empty ; | ||||||
| 
 | 
 | ||||||
| : ch>base64 ( ch -- ch ) | : ch>base64 ( ch -- ch ) | ||||||
|     "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/" nth ; |     "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/" | ||||||
|  |     nth ; inline | ||||||
| 
 | 
 | ||||||
| : base64>ch ( ch -- ch ) | : base64>ch ( ch -- ch ) | ||||||
|     { |     { | ||||||
|  | @ -19,32 +25,60 @@ IN: base64 | ||||||
|         f 0 f f f 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |         f 0 f f f 0 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 f f f f f f 26 27 28 29 30 31 32 33 34 35 36 37 38 39 |         22 23 24 25 f f f f f f 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 |         40 41 42 43 44 45 46 47 48 49 50 51 | ||||||
|     } nth ; |     } nth ; inline | ||||||
| 
 | 
 | ||||||
| : encode3 ( seq -- seq ) | SYMBOL: column | ||||||
|  | 
 | ||||||
|  | : write1-lines ( ch -- ) | ||||||
|  |     write1 | ||||||
|  |     column get [ | ||||||
|  |         1+ [ 76 = [ "\r\n" write ] when ] | ||||||
|  |         [ 76 mod column set ] bi | ||||||
|  |     ] when* ; | ||||||
|  | 
 | ||||||
|  | : write-lines ( str -- ) | ||||||
|  |     [ write1-lines ] each ; | ||||||
|  | 
 | ||||||
|  | : encode3 ( seq -- ) | ||||||
|     be> 4 <reversed> [ |     be> 4 <reversed> [ | ||||||
|         -6 * shift HEX: 3f bitand ch>base64 |         -6 * shift HEX: 3f bitand ch>base64 write1-lines | ||||||
|     ] with B{ } map-as ; |     ] with each ; inline | ||||||
| 
 | 
 | ||||||
| : decode4 ( str -- str ) | : encode-pad ( seq n -- ) | ||||||
|     0 [ base64>ch swap 6 shift bitor ] reduce 3 >be ; |     [ 3 0 pad-right binary [ encode3 ] with-byte-writer ] | ||||||
|  |     [ 1+ ] bi* head-slice 4 CHAR: = pad-right write-lines ; inline | ||||||
| 
 | 
 | ||||||
| : >base64-rem ( str -- str ) | ERROR: malformed-base64 ; | ||||||
|     [ 3 0 pad-right encode3 ] [ length 1+ ] bi | 
 | ||||||
|     head-slice 4 CHAR: = pad-right ; | : decode4 ( seq -- ) | ||||||
|  |     [ 0 [ base64>ch swap 6 shift bitor ] reduce 3 >be ] | ||||||
|  |     [ [ CHAR: = = ] count ] bi head-slice* | ||||||
|  |     [ write1 ] each ; inline | ||||||
| 
 | 
 | ||||||
| PRIVATE> | PRIVATE> | ||||||
| 
 | 
 | ||||||
| : >base64 ( seq -- base64 ) | : encode-base64 ( -- ) | ||||||
|     #! cut string into two pieces, convert 3 bytes at a time |     3 read dup length { | ||||||
|     #! pad string with = when not enough bits |         { 0 [ drop ] } | ||||||
|     dup length dup 3 mod - cut |         { 3 [ encode3 encode-base64 ] } | ||||||
|     [ 3 <groups> [ encode3 ] map concat ] |         [ encode-pad encode-base64 ] | ||||||
|     [ [ "" ] [ >base64-rem ] if-empty ] |     } case ; | ||||||
|     bi* append ; |  | ||||||
| 
 | 
 | ||||||
| : base64> ( base64 -- seq ) | : encode-base64-lines ( -- ) | ||||||
|     #! input length must be a multiple of 4 |     0 column [ encode-base64 ] with-variable ; | ||||||
|     [ 4 <groups> [ decode4 ] map concat ] | 
 | ||||||
|     [ [ CHAR: = = ] count-end ] | : decode-base64 ( -- ) | ||||||
|     bi head* ; |     "\n\r" 4 read-ignoring dup length { | ||||||
|  |         { 0 [ drop ] } | ||||||
|  |         { 4 [ decode4 decode-base64 ] } | ||||||
|  |         [ malformed-base64 ] | ||||||
|  |     } case ; | ||||||
|  | 
 | ||||||
|  | : >base64 ( str -- base64 ) | ||||||
|  |     binary [ [ encode-base64 ] with-string-reader ] with-byte-writer ; | ||||||
|  | 
 | ||||||
|  | : base64> ( base64 -- str ) | ||||||
|  |     [ binary [ decode-base64 ] with-byte-reader ] with-string-writer ; | ||||||
|  | 
 | ||||||
|  | : >base64-lines ( str -- base64 ) | ||||||
|  |     binary [ [ encode-base64-lines ] with-string-reader ] with-byte-writer ; | ||||||
		Loading…
	
		Reference in New Issue