2018-04-18 14:38:06 +00:00
|
|
|
/* gpgme.js - Javascript integration for gpgme
|
|
|
|
* Copyright (C) 2018 Bundesamt für Sicherheit in der Informationstechnik
|
|
|
|
*
|
|
|
|
* This file is part of GPGME.
|
|
|
|
*
|
|
|
|
* GPGME 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 2.1 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* GPGME 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 program; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
* SPDX-License-Identifier: LGPL-2.1+
|
2018-06-06 11:05:53 +00:00
|
|
|
*
|
|
|
|
* Author(s):
|
|
|
|
* Maximilian Krambach <mkrambach@intevation.de>
|
2018-04-18 14:38:06 +00:00
|
|
|
*/
|
2018-06-06 11:05:53 +00:00
|
|
|
|
|
|
|
import { gpgme_error } from './Errors';
|
2018-04-18 14:38:06 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Tries to return an array of fingerprints, either from input fingerprints or
|
2018-07-10 12:32:26 +00:00
|
|
|
* from Key objects (openpgp Keys or GPGME_Keys are both accepted).
|
|
|
|
*
|
|
|
|
* @param {Object | Array<Object> | String | Array<String>} input
|
|
|
|
* @returns {Array<String>} Array of fingerprints, or an empty array
|
2018-04-18 14:38:06 +00:00
|
|
|
*/
|
2018-08-20 13:12:01 +00:00
|
|
|
export function toKeyIdArray (input){
|
2018-04-18 14:38:06 +00:00
|
|
|
if (!input){
|
2018-04-25 08:54:24 +00:00
|
|
|
return [];
|
2018-04-23 15:18:46 +00:00
|
|
|
}
|
|
|
|
if (!Array.isArray(input)){
|
|
|
|
input = [input];
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
2018-04-23 15:18:46 +00:00
|
|
|
let result = [];
|
|
|
|
for (let i=0; i < input.length; i++){
|
2018-08-20 13:12:01 +00:00
|
|
|
if (typeof (input[i]) === 'string'){
|
2018-04-18 14:38:06 +00:00
|
|
|
if (isFingerprint(input[i]) === true){
|
|
|
|
result.push(input[i]);
|
|
|
|
} else {
|
2018-07-10 12:32:26 +00:00
|
|
|
// MSG_NOT_A_FPR is just a console warning if warning enabled
|
|
|
|
// in src/Errors.js
|
2018-04-25 13:59:36 +00:00
|
|
|
gpgme_error('MSG_NOT_A_FPR');
|
2018-04-23 15:18:46 +00:00
|
|
|
}
|
2018-08-20 13:12:01 +00:00
|
|
|
} else if (typeof (input[i]) === 'object'){
|
2018-04-23 15:18:46 +00:00
|
|
|
let fpr = '';
|
2018-08-17 12:40:27 +00:00
|
|
|
if (input[i].hasOwnProperty('fingerprint')){
|
2018-04-23 15:18:46 +00:00
|
|
|
fpr = input[i].fingerprint;
|
2018-04-25 17:45:39 +00:00
|
|
|
} else if (input[i].hasOwnProperty('primaryKey') &&
|
2018-05-04 10:56:59 +00:00
|
|
|
input[i].primaryKey.hasOwnProperty('getFingerprint')){
|
2018-06-06 11:05:53 +00:00
|
|
|
fpr = input[i].primaryKey.getFingerprint();
|
2018-04-23 15:18:46 +00:00
|
|
|
}
|
|
|
|
if (isFingerprint(fpr) === true){
|
|
|
|
result.push(fpr);
|
2018-04-25 08:54:24 +00:00
|
|
|
} else {
|
2018-04-25 13:59:36 +00:00
|
|
|
gpgme_error('MSG_NOT_A_FPR');
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
2018-04-23 15:18:46 +00:00
|
|
|
} else {
|
2018-04-25 13:59:36 +00:00
|
|
|
return gpgme_error('PARAM_WRONG');
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
2018-04-23 15:18:46 +00:00
|
|
|
}
|
|
|
|
if (result.length === 0){
|
2018-04-25 08:54:24 +00:00
|
|
|
return [];
|
2018-04-23 15:18:46 +00:00
|
|
|
} else {
|
2018-04-18 14:38:06 +00:00
|
|
|
return result;
|
|
|
|
}
|
2018-06-06 11:05:53 +00:00
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
|
|
|
|
/**
|
2018-07-10 12:32:26 +00:00
|
|
|
* Check if values are valid hexadecimal values of a specified length
|
|
|
|
* @param {String} key input value.
|
2018-04-18 14:38:06 +00:00
|
|
|
* @param {int} len the expected length of the value
|
2018-07-10 12:32:26 +00:00
|
|
|
* @returns {Boolean} true if value passes test
|
|
|
|
* @private
|
2018-04-18 14:38:06 +00:00
|
|
|
*/
|
2018-08-20 13:12:01 +00:00
|
|
|
function hextest (key, len){
|
|
|
|
if (!key || typeof (key) !== 'string'){
|
2018-04-18 14:38:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (key.length !== len){
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
let regexp= /^[0-9a-fA-F]*$/i;
|
|
|
|
return regexp.test(key);
|
2018-06-06 11:05:53 +00:00
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
|
|
|
|
/**
|
2018-07-10 12:32:26 +00:00
|
|
|
* check if the input is a valid Fingerprint
|
|
|
|
* (Hex string with a length of 40 characters)
|
|
|
|
* @param {String} value to check
|
|
|
|
* @returns {Boolean} true if value passes test
|
2018-04-18 14:38:06 +00:00
|
|
|
*/
|
2018-08-20 13:12:01 +00:00
|
|
|
export function isFingerprint (value){
|
2018-07-10 12:32:26 +00:00
|
|
|
return hextest(value, 40);
|
2018-06-06 11:05:53 +00:00
|
|
|
}
|
2018-05-25 17:02:18 +00:00
|
|
|
|
2018-04-23 15:18:46 +00:00
|
|
|
/**
|
2018-07-10 12:32:26 +00:00
|
|
|
* check if the input is a valid gnupg long ID (Hex string with a length of 16
|
|
|
|
* characters)
|
|
|
|
* @param {String} value to check
|
|
|
|
* @returns {Boolean} true if value passes test
|
2018-04-23 15:18:46 +00:00
|
|
|
*/
|
2018-08-20 13:12:01 +00:00
|
|
|
export function isLongId (value){
|
2018-07-10 12:32:26 +00:00
|
|
|
return hextest(value, 16);
|
2018-06-06 11:05:53 +00:00
|
|
|
}
|
2018-08-17 17:20:35 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively decodes input (utf8) to output (utf-16; javascript) strings
|
|
|
|
* @param {Object | Array | String} property
|
|
|
|
*/
|
2018-08-20 13:12:01 +00:00
|
|
|
export function decode (property){
|
2018-08-17 17:20:35 +00:00
|
|
|
if (typeof property === 'string'){
|
2018-08-27 11:00:50 +00:00
|
|
|
try {
|
2018-08-30 13:37:37 +00:00
|
|
|
return decodeURIComponent(escape(unescape(property)));
|
2018-08-27 11:00:50 +00:00
|
|
|
}
|
|
|
|
catch (error){
|
|
|
|
if (error instanceof URIError) {
|
|
|
|
return property;
|
|
|
|
}
|
|
|
|
}
|
2018-08-17 17:20:35 +00:00
|
|
|
} else if (Array.isArray(property)){
|
|
|
|
let res = [];
|
|
|
|
for (let arr=0; arr < property.length; arr++){
|
|
|
|
res.push(decode(property[arr]));
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
} else if (typeof property === 'object'){
|
|
|
|
const keys = Object.keys(property);
|
|
|
|
if (keys.length){
|
|
|
|
let res = {};
|
|
|
|
for (let k=0; k < keys.length; k++ ){
|
|
|
|
res[keys[k]] = decode(property[keys[k]]);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
return property;
|
|
|
|
}
|
|
|
|
return property;
|
2018-08-22 14:32:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Turns a base64 encoded string into an uint8 array
|
|
|
|
* @param {String} base64 encoded String
|
|
|
|
* @returns {Uint8Array}
|
|
|
|
* adapted from https://gist.github.com/borismus/1032746
|
|
|
|
*/
|
|
|
|
export function atobArray (base64) {
|
|
|
|
if (typeof (base64) !== 'string'){
|
|
|
|
throw gpgme_error('DECODE_FAIL');
|
|
|
|
}
|
|
|
|
const raw = window.atob(base64);
|
|
|
|
const rawLength = raw.length;
|
|
|
|
let array = new Uint8Array(new ArrayBuffer(rawLength));
|
|
|
|
for (let i = 0; i < rawLength; i++) {
|
|
|
|
array[i] = raw.charCodeAt(i);
|
|
|
|
}
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Turns a Uint8Array into an utf8-String
|
|
|
|
* @param {*} array Uint8Array
|
|
|
|
* @returns {String}
|
|
|
|
* Taken and slightly adapted from
|
|
|
|
* http://www.onicos.com/staff/iz/amuse/javascript/expert/utf.txt
|
|
|
|
* (original header:
|
|
|
|
* utf.js - UTF-8 <=> UTF-16 convertion
|
|
|
|
*
|
|
|
|
* Copyright (C) 1999 Masanao Izumo <iz@onicos.co.jp>
|
|
|
|
* Version: 1.0
|
|
|
|
* LastModified: Dec 25 1999
|
|
|
|
* This library is free. You can redistribute it and/or modify it.
|
|
|
|
* )
|
|
|
|
*/
|
|
|
|
export function Utf8ArrayToStr (array) {
|
|
|
|
let out, i, len, c, char2, char3;
|
|
|
|
out = '';
|
|
|
|
len = array.length;
|
|
|
|
i = 0;
|
|
|
|
if (array instanceof Uint8Array === false){
|
|
|
|
throw gpgme_error('DECODE_FAIL');
|
|
|
|
}
|
|
|
|
while (i < len) {
|
|
|
|
c = array[i++];
|
|
|
|
switch (c >> 4) {
|
|
|
|
case 0: case 1: case 2: case 3: case 4: case 5: case 6: case 7:
|
|
|
|
// 0xxxxxxx
|
|
|
|
out += String.fromCharCode(c);
|
|
|
|
break;
|
|
|
|
case 12: case 13:
|
|
|
|
// 110x xxxx 10xx xxxx
|
|
|
|
char2 = array[i++];
|
|
|
|
out += String.fromCharCode(((c & 0x1F) << 6) | (char2 & 0x3F));
|
|
|
|
break;
|
|
|
|
case 14:
|
|
|
|
// 1110 xxxx 10xx xxxx 10xx xxxx
|
|
|
|
char2 = array[i++];
|
|
|
|
char3 = array[i++];
|
|
|
|
out += String.fromCharCode(((c & 0x0F) << 12) |
|
|
|
|
((char2 & 0x3F) << 6) |
|
|
|
|
((char3 & 0x3F) << 0));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|