;;; guile-email --- Guile email parser ;;; Copyright © 2020 Arun Isaac <arunisaac@systemreboot.net> ;;; ;;; This file is part of guile-email. ;;; ;;; guile-email is free software; you can redistribute it and/or modify ;;; it under the terms of the GNU Affero General Public License as ;;; published by the Free Software Foundation; either version 3 of the ;;; License, or (at your option) any later version. ;;; ;;; guile-email is distributed in the hope that it will be useful, but ;;; WITHOUT ANY WARRANTY; without even the implied warranty of ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU ;;; Affero General Public License for more details. ;;; ;;; You should have received a copy of the GNU Affero General Public ;;; License along with guile-email. If not, see ;;; <http://www.gnu.org/licenses/>. (use-modules (email base64) (ice-9 iconv) (srfi srfi-64)) (load "encoding.scm") (define (string-has-only-valid-base64-characters? str) (string-every (char-set-union (char-set #\newline #\return #\space) (ucs-range->char-set (char->integer #\A) (1+ (char->integer #\Z))) (ucs-range->char-set (char->integer #\a) (1+ (char->integer #\z))) (ucs-range->char-set (char->integer #\0) (1+ (char->integer #\9))) (char-set #\+ #\/ #\=)) str)) (test-begin "base64") (let ((decoded-text "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.") (encoded-text "TWFuIGlzIGRpc3Rpbmd1aXNoZWQsIG5vdCBvbmx5IGJ5IGhpcyByZWFzb24sIGJ1dCBieSB0aGlz IHNpbmd1bGFyIHBhc3Npb24gZnJvbSBvdGhlciBhbmltYWxzLCB3aGljaCBpcyBhIGx1c3Qgb2Yg dGhlIG1pbmQsIHRoYXQgYnkgYSBwZXJzZXZlcmFuY2Ugb2YgZGVsaWdodCBpbiB0aGUgY29udGlu dWVkIGFuZCBpbmRlZmF0aWdhYmxlIGdlbmVyYXRpb24gb2Yga25vd2xlZGdlLCBleGNlZWRzIHRo ZSBzaG9ydCB2ZWhlbWVuY2Ugb2YgYW55IGNhcm5hbCBwbGVhc3VyZS4=") (charset "UTF-8")) (test-equal "base64 wikipedia example: decoding" (bytevector->string (base64-decode encoded-text) charset) decoded-text) (test-equal "base64 wikipedia example: encoding" (base64-encode (string->bytevector decoded-text charset)) encoded-text) (test-assert "base64 wikipedia example: encoded output should not be more than 76 columns wide" (each-line-has-a-maximum-of-76-characters? (base64-encode (string->bytevector decoded-text charset))))) (test-assert "base64 random bytevector: base64-encode and base64-decode are inverses of each other" (every (lambda (len) (let ((x (random-bytevector len))) (equal? x (base64-decode (base64-encode x))))) (iota 1000))) (test-assert "base64 random bytevector: encoded output should not be more than 76 columns wide" (every (lambda (len) (each-line-has-a-maximum-of-76-characters? (base64-encode (random-bytevector len)))) (iota 1000))) (test-assert "base64 random bytevector: encoded output must only consist of characters from the base64 alphabet" (every (lambda (len) (string-has-only-valid-base64-characters? (base64-encode (random-bytevector len)))) (iota 1000))) (test-equal "base64 decoding should ignore invalid characters" (bytevector->string (base64-decode "..TWFu,") "utf-8") "Man") (test-end "base64")