| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566 |
- // ASN.1 JavaScript decoder
- // Copyright (c) 2008-2014 Lapo Luchini <lapo@lapo.it>
- // Permission to use, copy, modify, and/or distribute this software for any
- // purpose with or without fee is hereby granted, provided that the above
- // copyright notice and this permission notice appear in all copies.
- //
- // THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
- // WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
- // MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
- // ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
- // WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
- // ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
- // OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
- /*jshint browser: true, strict: true, immed: true, latedef: true, undef: true, regexdash: false */
- /*global oids */
- import { Int10 } from "./int10";
- var ellipsis = "\u2026";
- var reTimeS = /^(\d\d)(0[1-9]|1[0-2])(0[1-9]|[12]\d|3[01])([01]\d|2[0-3])(?:([0-5]\d)(?:([0-5]\d)(?:[.,](\d{1,3}))?)?)?(Z|[-+](?:[0]\d|1[0-2])([0-5]\d)?)?$/;
- var reTimeL = /^(\d\d\d\d)(0[1-9]|1[0-2])(0[1-9]|[12]\d|3[01])([01]\d|2[0-3])(?:([0-5]\d)(?:([0-5]\d)(?:[.,](\d{1,3}))?)?)?(Z|[-+](?:[0]\d|1[0-2])([0-5]\d)?)?$/;
- function stringCut(str, len) {
- if (str.length > len) {
- str = str.substring(0, len) + ellipsis;
- }
- return str;
- }
- var Stream = /** @class */ (function () {
- function Stream(enc, pos) {
- this.hexDigits = "0123456789ABCDEF";
- if (enc instanceof Stream) {
- this.enc = enc.enc;
- this.pos = enc.pos;
- }
- else {
- // enc should be an array or a binary string
- this.enc = enc;
- this.pos = pos;
- }
- }
- Stream.prototype.get = function (pos) {
- if (pos === undefined) {
- pos = this.pos++;
- }
- if (pos >= this.enc.length) {
- throw new Error("Requesting byte offset ".concat(pos, " on a stream of length ").concat(this.enc.length));
- }
- return ("string" === typeof this.enc) ? this.enc.charCodeAt(pos) : this.enc[pos];
- };
- Stream.prototype.hexByte = function (b) {
- return this.hexDigits.charAt((b >> 4) & 0xF) + this.hexDigits.charAt(b & 0xF);
- };
- Stream.prototype.hexDump = function (start, end, raw) {
- var s = "";
- for (var i = start; i < end; ++i) {
- s += this.hexByte(this.get(i));
- if (raw !== true) {
- switch (i & 0xF) {
- case 0x7:
- s += " ";
- break;
- case 0xF:
- s += "\n";
- break;
- default:
- s += " ";
- }
- }
- }
- return s;
- };
- Stream.prototype.isASCII = function (start, end) {
- for (var i = start; i < end; ++i) {
- var c = this.get(i);
- if (c < 32 || c > 176) {
- return false;
- }
- }
- return true;
- };
- Stream.prototype.parseStringISO = function (start, end) {
- var s = "";
- for (var i = start; i < end; ++i) {
- s += String.fromCharCode(this.get(i));
- }
- return s;
- };
- Stream.prototype.parseStringUTF = function (start, end) {
- var s = "";
- for (var i = start; i < end;) {
- var c = this.get(i++);
- if (c < 128) {
- s += String.fromCharCode(c);
- }
- else if ((c > 191) && (c < 224)) {
- s += String.fromCharCode(((c & 0x1F) << 6) | (this.get(i++) & 0x3F));
- }
- else {
- s += String.fromCharCode(((c & 0x0F) << 12) | ((this.get(i++) & 0x3F) << 6) | (this.get(i++) & 0x3F));
- }
- }
- return s;
- };
- Stream.prototype.parseStringBMP = function (start, end) {
- var str = "";
- var hi;
- var lo;
- for (var i = start; i < end;) {
- hi = this.get(i++);
- lo = this.get(i++);
- str += String.fromCharCode((hi << 8) | lo);
- }
- return str;
- };
- Stream.prototype.parseTime = function (start, end, shortYear) {
- var s = this.parseStringISO(start, end);
- var m = (shortYear ? reTimeS : reTimeL).exec(s);
- if (!m) {
- return "Unrecognized time: " + s;
- }
- if (shortYear) {
- // to avoid querying the timer, use the fixed range [1970, 2069]
- // it will conform with ITU X.400 [-10, +40] sliding window until 2030
- m[1] = +m[1];
- m[1] += (+m[1] < 70) ? 2000 : 1900;
- }
- s = m[1] + "-" + m[2] + "-" + m[3] + " " + m[4];
- if (m[5]) {
- s += ":" + m[5];
- if (m[6]) {
- s += ":" + m[6];
- if (m[7]) {
- s += "." + m[7];
- }
- }
- }
- if (m[8]) {
- s += " UTC";
- if (m[8] != "Z") {
- s += m[8];
- if (m[9]) {
- s += ":" + m[9];
- }
- }
- }
- return s;
- };
- Stream.prototype.parseInteger = function (start, end) {
- var v = this.get(start);
- var neg = (v > 127);
- var pad = neg ? 255 : 0;
- var len;
- var s = "";
- // skip unuseful bits (not allowed in DER)
- while (v == pad && ++start < end) {
- v = this.get(start);
- }
- len = end - start;
- if (len === 0) {
- return neg ? -1 : 0;
- }
- // show bit length of huge integers
- if (len > 4) {
- s = v;
- len <<= 3;
- while (((+s ^ pad) & 0x80) == 0) {
- s = +s << 1;
- --len;
- }
- s = "(" + len + " bit)\n";
- }
- // decode the integer
- if (neg) {
- v = v - 256;
- }
- var n = new Int10(v);
- for (var i = start + 1; i < end; ++i) {
- n.mulAdd(256, this.get(i));
- }
- return s + n.toString();
- };
- Stream.prototype.parseBitString = function (start, end, maxLength) {
- var unusedBit = this.get(start);
- var lenBit = ((end - start - 1) << 3) - unusedBit;
- var intro = "(" + lenBit + " bit)\n";
- var s = "";
- for (var i = start + 1; i < end; ++i) {
- var b = this.get(i);
- var skip = (i == end - 1) ? unusedBit : 0;
- for (var j = 7; j >= skip; --j) {
- s += (b >> j) & 1 ? "1" : "0";
- }
- if (s.length > maxLength) {
- return intro + stringCut(s, maxLength);
- }
- }
- return intro + s;
- };
- Stream.prototype.parseOctetString = function (start, end, maxLength) {
- if (this.isASCII(start, end)) {
- return stringCut(this.parseStringISO(start, end), maxLength);
- }
- var len = end - start;
- var s = "(" + len + " byte)\n";
- maxLength /= 2; // we work in bytes
- if (len > maxLength) {
- end = start + maxLength;
- }
- for (var i = start; i < end; ++i) {
- s += this.hexByte(this.get(i));
- }
- if (len > maxLength) {
- s += ellipsis;
- }
- return s;
- };
- Stream.prototype.parseOID = function (start, end, maxLength) {
- var s = "";
- var n = new Int10();
- var bits = 0;
- for (var i = start; i < end; ++i) {
- var v = this.get(i);
- n.mulAdd(128, v & 0x7F);
- bits += 7;
- if (!(v & 0x80)) { // finished
- if (s === "") {
- n = n.simplify();
- if (n instanceof Int10) {
- n.sub(80);
- s = "2." + n.toString();
- }
- else {
- var m = n < 80 ? n < 40 ? 0 : 1 : 2;
- s = m + "." + (n - m * 40);
- }
- }
- else {
- s += "." + n.toString();
- }
- if (s.length > maxLength) {
- return stringCut(s, maxLength);
- }
- n = new Int10();
- bits = 0;
- }
- }
- if (bits > 0) {
- s += ".incomplete";
- }
- return s;
- };
- return Stream;
- }());
- export { Stream };
- var ASN1 = /** @class */ (function () {
- function ASN1(stream, header, length, tag, sub) {
- if (!(tag instanceof ASN1Tag)) {
- throw new Error("Invalid tag value.");
- }
- this.stream = stream;
- this.header = header;
- this.length = length;
- this.tag = tag;
- this.sub = sub;
- }
- ASN1.prototype.typeName = function () {
- switch (this.tag.tagClass) {
- case 0: // universal
- switch (this.tag.tagNumber) {
- case 0x00:
- return "EOC";
- case 0x01:
- return "BOOLEAN";
- case 0x02:
- return "INTEGER";
- case 0x03:
- return "BIT_STRING";
- case 0x04:
- return "OCTET_STRING";
- case 0x05:
- return "NULL";
- case 0x06:
- return "OBJECT_IDENTIFIER";
- case 0x07:
- return "ObjectDescriptor";
- case 0x08:
- return "EXTERNAL";
- case 0x09:
- return "REAL";
- case 0x0A:
- return "ENUMERATED";
- case 0x0B:
- return "EMBEDDED_PDV";
- case 0x0C:
- return "UTF8String";
- case 0x10:
- return "SEQUENCE";
- case 0x11:
- return "SET";
- case 0x12:
- return "NumericString";
- case 0x13:
- return "PrintableString"; // ASCII subset
- case 0x14:
- return "TeletexString"; // aka T61String
- case 0x15:
- return "VideotexString";
- case 0x16:
- return "IA5String"; // ASCII
- case 0x17:
- return "UTCTime";
- case 0x18:
- return "GeneralizedTime";
- case 0x19:
- return "GraphicString";
- case 0x1A:
- return "VisibleString"; // ASCII subset
- case 0x1B:
- return "GeneralString";
- case 0x1C:
- return "UniversalString";
- case 0x1E:
- return "BMPString";
- }
- return "Universal_" + this.tag.tagNumber.toString();
- case 1:
- return "Application_" + this.tag.tagNumber.toString();
- case 2:
- return "[" + this.tag.tagNumber.toString() + "]"; // Context
- case 3:
- return "Private_" + this.tag.tagNumber.toString();
- }
- };
- ASN1.prototype.content = function (maxLength) {
- if (this.tag === undefined) {
- return null;
- }
- if (maxLength === undefined) {
- maxLength = Infinity;
- }
- var content = this.posContent();
- var len = Math.abs(this.length);
- if (!this.tag.isUniversal()) {
- if (this.sub !== null) {
- return "(" + this.sub.length + " elem)";
- }
- return this.stream.parseOctetString(content, content + len, maxLength);
- }
- switch (this.tag.tagNumber) {
- case 0x01: // BOOLEAN
- return (this.stream.get(content) === 0) ? "false" : "true";
- case 0x02: // INTEGER
- return this.stream.parseInteger(content, content + len);
- case 0x03: // BIT_STRING
- return this.sub ? "(" + this.sub.length + " elem)" :
- this.stream.parseBitString(content, content + len, maxLength);
- case 0x04: // OCTET_STRING
- return this.sub ? "(" + this.sub.length + " elem)" :
- this.stream.parseOctetString(content, content + len, maxLength);
- // case 0x05: // NULL
- case 0x06: // OBJECT_IDENTIFIER
- return this.stream.parseOID(content, content + len, maxLength);
- // case 0x07: // ObjectDescriptor
- // case 0x08: // EXTERNAL
- // case 0x09: // REAL
- // case 0x0A: // ENUMERATED
- // case 0x0B: // EMBEDDED_PDV
- case 0x10: // SEQUENCE
- case 0x11: // SET
- if (this.sub !== null) {
- return "(" + this.sub.length + " elem)";
- }
- else {
- return "(no elem)";
- }
- case 0x0C: // UTF8String
- return stringCut(this.stream.parseStringUTF(content, content + len), maxLength);
- case 0x12: // NumericString
- case 0x13: // PrintableString
- case 0x14: // TeletexString
- case 0x15: // VideotexString
- case 0x16: // IA5String
- // case 0x19: // GraphicString
- case 0x1A: // VisibleString
- // case 0x1B: // GeneralString
- // case 0x1C: // UniversalString
- return stringCut(this.stream.parseStringISO(content, content + len), maxLength);
- case 0x1E: // BMPString
- return stringCut(this.stream.parseStringBMP(content, content + len), maxLength);
- case 0x17: // UTCTime
- case 0x18: // GeneralizedTime
- return this.stream.parseTime(content, content + len, (this.tag.tagNumber == 0x17));
- }
- return null;
- };
- ASN1.prototype.toString = function () {
- return this.typeName() + "@" + this.stream.pos + "[header:" + this.header + ",length:" + this.length + ",sub:" + ((this.sub === null) ? "null" : this.sub.length) + "]";
- };
- ASN1.prototype.toPrettyString = function (indent) {
- if (indent === undefined) {
- indent = "";
- }
- var s = indent + this.typeName() + " @" + this.stream.pos;
- if (this.length >= 0) {
- s += "+";
- }
- s += this.length;
- if (this.tag.tagConstructed) {
- s += " (constructed)";
- }
- else if ((this.tag.isUniversal() && ((this.tag.tagNumber == 0x03) || (this.tag.tagNumber == 0x04))) && (this.sub !== null)) {
- s += " (encapsulates)";
- }
- s += "\n";
- if (this.sub !== null) {
- indent += " ";
- for (var i = 0, max = this.sub.length; i < max; ++i) {
- s += this.sub[i].toPrettyString(indent);
- }
- }
- return s;
- };
- ASN1.prototype.posStart = function () {
- return this.stream.pos;
- };
- ASN1.prototype.posContent = function () {
- return this.stream.pos + this.header;
- };
- ASN1.prototype.posEnd = function () {
- return this.stream.pos + this.header + Math.abs(this.length);
- };
- ASN1.prototype.toHexString = function () {
- return this.stream.hexDump(this.posStart(), this.posEnd(), true);
- };
- ASN1.decodeLength = function (stream) {
- var buf = stream.get();
- var len = buf & 0x7F;
- if (len == buf) {
- return len;
- }
- // no reason to use Int10, as it would be a huge buffer anyways
- if (len > 6) {
- throw new Error("Length over 48 bits not supported at position " + (stream.pos - 1));
- }
- if (len === 0) {
- return null;
- } // undefined
- buf = 0;
- for (var i = 0; i < len; ++i) {
- buf = (buf * 256) + stream.get();
- }
- return buf;
- };
- /**
- * Retrieve the hexadecimal value (as a string) of the current ASN.1 element
- * @returns {string}
- * @public
- */
- ASN1.prototype.getHexStringValue = function () {
- var hexString = this.toHexString();
- var offset = this.header * 2;
- var length = this.length * 2;
- return hexString.substr(offset, length);
- };
- ASN1.decode = function (str) {
- var stream;
- if (!(str instanceof Stream)) {
- stream = new Stream(str, 0);
- }
- else {
- stream = str;
- }
- var streamStart = new Stream(stream);
- var tag = new ASN1Tag(stream);
- var len = ASN1.decodeLength(stream);
- var start = stream.pos;
- var header = start - streamStart.pos;
- var sub = null;
- var getSub = function () {
- var ret = [];
- if (len !== null) {
- // definite length
- var end = start + len;
- while (stream.pos < end) {
- ret[ret.length] = ASN1.decode(stream);
- }
- if (stream.pos != end) {
- throw new Error("Content size is not correct for container starting at offset " + start);
- }
- }
- else {
- // undefined length
- try {
- for (;;) {
- var s = ASN1.decode(stream);
- if (s.tag.isEOC()) {
- break;
- }
- ret[ret.length] = s;
- }
- len = start - stream.pos; // undefined lengths are represented as negative values
- }
- catch (e) {
- throw new Error("Exception while decoding undefined length content: " + e);
- }
- }
- return ret;
- };
- if (tag.tagConstructed) {
- // must have valid content
- sub = getSub();
- }
- else if (tag.isUniversal() && ((tag.tagNumber == 0x03) || (tag.tagNumber == 0x04))) {
- // sometimes BitString and OctetString are used to encapsulate ASN.1
- try {
- if (tag.tagNumber == 0x03) {
- if (stream.get() != 0) {
- throw new Error("BIT STRINGs with unused bits cannot encapsulate.");
- }
- }
- sub = getSub();
- for (var i = 0; i < sub.length; ++i) {
- if (sub[i].tag.isEOC()) {
- throw new Error("EOC is not supposed to be actual content.");
- }
- }
- }
- catch (e) {
- // but silently ignore when they don't
- sub = null;
- }
- }
- if (sub === null) {
- if (len === null) {
- throw new Error("We can't skip over an invalid tag with undefined length at offset " + start);
- }
- stream.pos = start + Math.abs(len);
- }
- return new ASN1(streamStart, header, len, tag, sub);
- };
- return ASN1;
- }());
- export { ASN1 };
- var ASN1Tag = /** @class */ (function () {
- function ASN1Tag(stream) {
- var buf = stream.get();
- this.tagClass = buf >> 6;
- this.tagConstructed = ((buf & 0x20) !== 0);
- this.tagNumber = buf & 0x1F;
- if (this.tagNumber == 0x1F) { // long tag
- var n = new Int10();
- do {
- buf = stream.get();
- n.mulAdd(128, buf & 0x7F);
- } while (buf & 0x80);
- this.tagNumber = n.simplify();
- }
- }
- ASN1Tag.prototype.isUniversal = function () {
- return this.tagClass === 0x00;
- };
- ASN1Tag.prototype.isEOC = function () {
- return this.tagClass === 0x00 && this.tagNumber === 0x00;
- };
- return ASN1Tag;
- }());
- export { ASN1Tag };
|