// GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // // This library is free software; you can redistribute it and/or // modify it under the terms of the GNU Lesser General Public // License as published by the Free Software Foundation; either // version 3.0 of the License, or (at your option) any later version. // // This library 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 // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, write to the Free Software // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA /** * @requires enums * @requires stream * @requires util */ import enums from '../enums'; import stream from '../stream'; import util from '../util'; /** * Implementation of the Literal Data Packet (Tag 11) * * {@link https://tools.ietf.org/html/rfc4880#section-5.9|RFC4880 5.9}: * A Literal Data packet contains the body of a message; data that is not to be * further interpreted. * @param {Date} date the creation date of the literal package * @memberof module:packet * @constructor */ function Literal(date=new Date()) { this.tag = enums.packet.literal; this.format = 'utf8'; // default format for literal data packets this.date = util.normalizeDate(date); this.text = null; // textual data representation this.data = null; // literal data representation this.filename = 'msg.txt'; } /** * Set the packet data to a javascript native string, end of line * will be normalized to \r\n and by default text is converted to UTF8 * @param {String | ReadableStream} text Any native javascript string * @param {utf8|binary|text|mime} format (optional) The format of the string of bytes */ Literal.prototype.setText = function(text, format='utf8') { this.format = format; this.text = text; this.data = null; }; /** * Returns literal data packets as native JavaScript string * with normalized end of line to \n * @param {Boolean} clone (optional) Whether to return a clone so that getBytes/getText can be called again * @returns {String | ReadableStream} literal data as text */ Literal.prototype.getText = function(clone=false) { if (this.text === null || util.isStream(this.text)) { // Assume that this.text has been read let lastChar = ''; const decoder = new TextDecoder('utf-8'); // eslint-disable-next-line no-inner-declarations function process(value, lastChunk=false) { // decode UTF8 const text = lastChar + decoder.decode(value, { stream: !lastChunk }); // normalize EOL to \n const normalized = util.nativeEOL(text); // if last char is \r, store it for the next chunk so we can normalize \r\n if (normalized[normalized.length - 1] === '\r') { lastChar = '\r'; return normalized.slice(0, -1); } lastChar = ''; return normalized; } this.text = stream.transform(this.getBytes(clone), process, () => process(new Uint8Array(), true)); } return this.text; }; /** * Set the packet data to value represented by the provided string of bytes. * @param {Uint8Array | ReadableStream} bytes The string of bytes * @param {utf8|binary|text|mime} format The format of the string of bytes */ Literal.prototype.setBytes = function(bytes, format) { this.format = format; this.data = bytes; this.text = null; }; /** * Get the byte sequence representing the literal packet data * @param {Boolean} clone (optional) Whether to return a clone so that getBytes/getText can be called again * @returns {Uint8Array | ReadableStream} A sequence of bytes */ Literal.prototype.getBytes = function(clone=false) { if (this.data === null) { // normalize EOL to \r\n const text = util.canonicalizeEOL(this.text); // encode UTF8 this.data = util.str_to_Uint8Array(util.encode_utf8(text)); } if (clone) { return stream.passiveClone(this.data); } return this.data; }; /** * Sets the filename of the literal packet data * @param {String} filename Any native javascript string */ Literal.prototype.setFilename = function(filename) { this.filename = filename; }; /** * Get the filename of the literal packet data * @returns {String} filename */ Literal.prototype.getFilename = function() { return this.filename; }; /** * Parsing function for a literal data packet (tag 11). * * @param {Uint8Array | ReadableStream} input Payload of a tag 11 packet * @returns {module:packet.Literal} object representation */ Literal.prototype.read = async function(bytes) { await stream.parse(bytes, async reader => { // - A one-octet field that describes how the data is formatted. const format = enums.read(enums.literal, await reader.readByte()); const filename_len = await reader.readByte(); this.filename = util.decode_utf8(util.Uint8Array_to_str(await reader.readBytes(filename_len))); this.date = util.readDate(await reader.readBytes(4)); const data = reader.remainder(); this.setBytes(data, format); }); }; /** * Creates a string representation of the packet * * @returns {Uint8Array | ReadableStream} Uint8Array representation of the packet */ Literal.prototype.write = function() { const filename = util.str_to_Uint8Array(util.encode_utf8(this.filename)); const filename_length = new Uint8Array([filename.length]); const format = new Uint8Array([enums.write(enums.literal, this.format)]); const date = util.writeDate(this.date); const data = this.getBytes(); return util.concat([format, filename_length, filename, date, data]); }; export default Literal;