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
|
|
|
/* global chrome */
|
|
|
|
|
|
|
|
import { permittedOperations } from './permittedOperations';
|
|
|
|
import { gpgme_error } from './Errors';
|
|
|
|
import { GPGME_Message, createMessage } from './Message';
|
2018-04-10 09:33:14 +00:00
|
|
|
|
2018-04-23 15:18:46 +00:00
|
|
|
/**
|
|
|
|
* A Connection handles the nativeMessaging interaction.
|
|
|
|
*/
|
2018-04-18 14:38:06 +00:00
|
|
|
export class Connection{
|
2018-04-10 09:33:14 +00:00
|
|
|
|
2018-04-18 14:38:06 +00:00
|
|
|
constructor(){
|
2018-04-23 15:18:46 +00:00
|
|
|
this.connect();
|
2018-04-24 16:44:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-05-25 09:53:24 +00:00
|
|
|
* Retrieves the information about the backend.
|
|
|
|
* @param {Boolean} details (optional) If set to false, the promise will
|
|
|
|
* just return a connection status
|
2018-07-04 11:38:54 +00:00
|
|
|
* @returns {Promise<Object>} result
|
|
|
|
* @returns {String} result.gpgme Version number of gpgme
|
|
|
|
* @returns {Array<Object>} result.info Further information about the
|
|
|
|
* backends.
|
2018-05-25 09:53:24 +00:00
|
|
|
* Example:
|
|
|
|
* "protocol": "OpenPGP",
|
|
|
|
* "fname": "/usr/bin/gpg",
|
|
|
|
* "version": "2.2.6",
|
|
|
|
* "req_version": "1.4.0",
|
|
|
|
* "homedir": "default"
|
2018-04-24 16:44:30 +00:00
|
|
|
*/
|
2018-05-25 09:53:24 +00:00
|
|
|
checkConnection(details = true){
|
|
|
|
if (details === true) {
|
|
|
|
return this.post(createMessage('version'));
|
|
|
|
} else {
|
|
|
|
let me = this;
|
2018-06-06 11:05:53 +00:00
|
|
|
return new Promise(function(resolve) {
|
2018-05-25 09:53:24 +00:00
|
|
|
Promise.race([
|
|
|
|
me.post(createMessage('version')),
|
|
|
|
new Promise(function(resolve, reject){
|
|
|
|
setTimeout(function(){
|
|
|
|
reject(gpgme_error('CONN_TIMEOUT'));
|
|
|
|
}, 500);
|
|
|
|
})
|
2018-06-06 11:05:53 +00:00
|
|
|
]).then(function(){ // success
|
|
|
|
resolve(true);
|
|
|
|
}, function(){ // failure
|
2018-05-25 09:53:24 +00:00
|
|
|
resolve(false);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
2018-04-10 09:33:14 +00:00
|
|
|
|
|
|
|
/**
|
2018-04-23 15:18:46 +00:00
|
|
|
* Immediately closes the open port.
|
2018-04-10 09:33:14 +00:00
|
|
|
*/
|
2018-04-18 14:38:06 +00:00
|
|
|
disconnect() {
|
|
|
|
if (this._connection){
|
|
|
|
this._connection.disconnect();
|
2018-05-25 09:53:24 +00:00
|
|
|
this._connection = null;
|
2018-04-10 09:33:14 +00:00
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
2018-04-10 09:33:14 +00:00
|
|
|
|
2018-04-23 15:18:46 +00:00
|
|
|
/**
|
|
|
|
* Opens a nativeMessaging port.
|
|
|
|
*/
|
|
|
|
connect(){
|
2018-05-25 09:53:24 +00:00
|
|
|
if (!this._connection){
|
2018-04-25 08:54:24 +00:00
|
|
|
this._connection = chrome.runtime.connectNative('gpgmejson');
|
2018-04-23 15:18:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-18 14:38:06 +00:00
|
|
|
/**
|
|
|
|
* Sends a message and resolves with the answer.
|
|
|
|
* @param {GPGME_Message} message
|
|
|
|
* @returns {Promise<Object>} the gnupg answer, or rejection with error
|
2018-04-23 15:18:46 +00:00
|
|
|
* information.
|
2018-04-18 14:38:06 +00:00
|
|
|
*/
|
|
|
|
post(message){
|
2018-06-06 11:05:53 +00:00
|
|
|
if (!message || !(message instanceof GPGME_Message)){
|
2018-05-28 14:52:50 +00:00
|
|
|
this.disconnect();
|
2018-06-06 11:05:53 +00:00
|
|
|
return Promise.reject(gpgme_error(
|
|
|
|
'PARAM_WRONG', 'Connection.post'));
|
2018-04-23 15:18:46 +00:00
|
|
|
}
|
|
|
|
if (message.isComplete !== true){
|
2018-05-28 14:52:50 +00:00
|
|
|
this.disconnect();
|
2018-04-25 13:59:36 +00:00
|
|
|
return Promise.reject(gpgme_error('MSG_INCOMPLETE'));
|
2018-04-10 09:33:14 +00:00
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
let me = this;
|
2018-06-08 15:54:58 +00:00
|
|
|
let chunksize = message.chunksize;
|
2018-04-10 09:33:14 +00:00
|
|
|
return new Promise(function(resolve, reject){
|
2018-05-22 12:24:16 +00:00
|
|
|
let answer = new Answer(message);
|
2018-04-18 14:38:06 +00:00
|
|
|
let listener = function(msg) {
|
2018-04-10 09:33:14 +00:00
|
|
|
if (!msg){
|
2018-06-06 11:05:53 +00:00
|
|
|
me._connection.onMessage.removeListener(listener);
|
2018-05-28 14:52:50 +00:00
|
|
|
me._connection.disconnect();
|
2018-04-25 13:59:36 +00:00
|
|
|
reject(gpgme_error('CONN_EMPTY_GPG_ANSWER'));
|
2018-04-18 14:38:06 +00:00
|
|
|
} else {
|
2018-06-08 15:54:58 +00:00
|
|
|
let answer_result = answer.collect(msg);
|
2018-04-25 08:54:24 +00:00
|
|
|
if (answer_result !== true){
|
2018-05-07 16:27:25 +00:00
|
|
|
me._connection.onMessage.removeListener(listener);
|
2018-05-28 14:52:50 +00:00
|
|
|
me._connection.disconnect();
|
2018-04-25 08:54:24 +00:00
|
|
|
reject(answer_result);
|
2018-04-18 14:38:06 +00:00
|
|
|
} else {
|
2018-06-08 15:54:58 +00:00
|
|
|
if (msg.more === true){
|
|
|
|
me._connection.postMessage({
|
|
|
|
'op': 'getmore',
|
|
|
|
'chunksize': chunksize
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
me._connection.onMessage.removeListener(listener);
|
|
|
|
me._connection.disconnect();
|
|
|
|
if (answer.message instanceof Error){
|
|
|
|
reject(answer.message);
|
|
|
|
} else {
|
|
|
|
resolve(answer.message);
|
|
|
|
}
|
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
2018-04-10 09:33:14 +00:00
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
};
|
|
|
|
me._connection.onMessage.addListener(listener);
|
2018-04-25 09:32:21 +00:00
|
|
|
if (permittedOperations[message.operation].pinentry){
|
|
|
|
return me._connection.postMessage(message.message);
|
|
|
|
} else {
|
2018-04-25 13:59:36 +00:00
|
|
|
return Promise.race([
|
|
|
|
me._connection.postMessage(message.message),
|
|
|
|
function(resolve, reject){
|
|
|
|
setTimeout(function(){
|
2018-05-28 14:52:50 +00:00
|
|
|
me._connection.disconnect();
|
2018-04-25 13:59:36 +00:00
|
|
|
reject(gpgme_error('CONN_TIMEOUT'));
|
|
|
|
}, 5000);
|
|
|
|
}]).then(function(result){
|
2018-06-06 11:05:53 +00:00
|
|
|
return result;
|
2018-05-07 16:27:25 +00:00
|
|
|
}, function(reject){
|
2018-06-06 11:05:53 +00:00
|
|
|
if(!(reject instanceof Error)) {
|
2018-05-28 14:52:50 +00:00
|
|
|
me._connection.disconnect();
|
2018-05-07 16:27:25 +00:00
|
|
|
return gpgme_error('GNUPG_ERROR', reject);
|
|
|
|
} else {
|
|
|
|
return reject;
|
|
|
|
}
|
2018-04-25 13:59:36 +00:00
|
|
|
});
|
2018-04-25 09:32:21 +00:00
|
|
|
}
|
2018-04-10 09:33:14 +00:00
|
|
|
});
|
2018-06-06 11:05:53 +00:00
|
|
|
}
|
|
|
|
}
|
2018-04-10 09:33:14 +00:00
|
|
|
|
2018-04-18 14:38:06 +00:00
|
|
|
/**
|
|
|
|
* A class for answer objects, checking and processing the return messages of
|
2018-04-23 15:18:46 +00:00
|
|
|
* the nativeMessaging communication.
|
2018-06-06 11:05:53 +00:00
|
|
|
* @param {String} operation The operation, to look up validity of returning
|
|
|
|
* messages
|
2018-04-18 14:38:06 +00:00
|
|
|
*/
|
|
|
|
class Answer{
|
2018-04-10 09:33:14 +00:00
|
|
|
|
2018-05-22 12:24:16 +00:00
|
|
|
constructor(message){
|
|
|
|
this.operation = message.operation;
|
2018-06-08 15:54:58 +00:00
|
|
|
this.expect = message.expect;
|
2018-04-10 09:33:14 +00:00
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
|
2018-06-08 15:54:58 +00:00
|
|
|
collect(msg){
|
|
|
|
if (typeof(msg) !== 'object' || !msg.hasOwnProperty('response')) {
|
|
|
|
return gpgme_error('CONN_UNEXPECTED_ANSWER');
|
|
|
|
}
|
|
|
|
if (this._responseb64 === undefined){
|
|
|
|
//this._responseb64 = [msg.response];
|
|
|
|
this._responseb64 = msg.response;
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
//this._responseb64.push(msg.response);
|
|
|
|
this._responseb64 += msg.response;
|
|
|
|
return true;
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
2018-06-08 15:54:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
get message(){
|
|
|
|
if (this._responseb64 === undefined){
|
|
|
|
return gpgme_error('CONN_UNEXPECTED_ANSWER');
|
|
|
|
}
|
|
|
|
// let _decodedResponse = JSON.parse(atob(this._responseb64.join('')));
|
|
|
|
let _decodedResponse = JSON.parse(atob(this._responseb64));
|
|
|
|
let _response = {};
|
|
|
|
let messageKeys = Object.keys(_decodedResponse);
|
2018-04-18 14:38:06 +00:00
|
|
|
let poa = permittedOperations[this.operation].answer;
|
2018-04-25 08:54:24 +00:00
|
|
|
if (messageKeys.length === 0){
|
2018-04-25 13:59:36 +00:00
|
|
|
return gpgme_error('CONN_UNEXPECTED_ANSWER');
|
2018-04-25 08:54:24 +00:00
|
|
|
}
|
2018-04-18 14:38:06 +00:00
|
|
|
for (let i= 0; i < messageKeys.length; i++){
|
|
|
|
let key = messageKeys[i];
|
|
|
|
switch (key) {
|
2018-06-06 11:05:53 +00:00
|
|
|
case 'type':
|
2018-06-08 15:54:58 +00:00
|
|
|
if (_decodedResponse.type === 'error'){
|
|
|
|
return (gpgme_error('GNUPG_ERROR', _decodedResponse.msg));
|
|
|
|
} else if (poa.type.indexOf(_decodedResponse.type) < 0){
|
2018-06-06 11:05:53 +00:00
|
|
|
return gpgme_error('CONN_UNEXPECTED_ANSWER');
|
|
|
|
}
|
|
|
|
break;
|
2018-06-08 15:54:58 +00:00
|
|
|
case 'base64':
|
2018-06-06 11:05:53 +00:00
|
|
|
break;
|
2018-06-08 15:54:58 +00:00
|
|
|
case 'msg':
|
|
|
|
if (_decodedResponse.type === 'error'){
|
|
|
|
return (gpgme_error('GNUPG_ERROR', _decodedResponse.msg));
|
2018-06-06 11:05:53 +00:00
|
|
|
}
|
2018-06-08 15:54:58 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (!poa.data.hasOwnProperty(key)){
|
|
|
|
return gpgme_error('CONN_UNEXPECTED_ANSWER');
|
2018-06-06 11:05:53 +00:00
|
|
|
}
|
2018-06-08 15:54:58 +00:00
|
|
|
if( typeof(_decodedResponse[key]) !== poa.data[key] ){
|
2018-06-06 11:05:53 +00:00
|
|
|
return gpgme_error('CONN_UNEXPECTED_ANSWER');
|
|
|
|
}
|
2018-06-08 15:54:58 +00:00
|
|
|
if (_decodedResponse.base64 === true
|
|
|
|
&& poa.data[key] === 'string'
|
|
|
|
&& this.expect === undefined
|
|
|
|
){
|
|
|
|
_response[key] = decodeURIComponent(
|
|
|
|
atob(_decodedResponse[key]).split('').map(
|
2018-06-06 11:05:53 +00:00
|
|
|
function(c) {
|
|
|
|
return '%' +
|
2018-06-08 15:54:58 +00:00
|
|
|
('00' + c.charCodeAt(0).toString(16)).slice(-2);
|
2018-06-06 11:05:53 +00:00
|
|
|
}).join(''));
|
2018-06-08 15:54:58 +00:00
|
|
|
} else {
|
|
|
|
_response[key] = _decodedResponse[key];
|
2018-05-09 17:40:57 +00:00
|
|
|
}
|
2018-06-08 15:54:58 +00:00
|
|
|
break;
|
2018-05-09 17:40:57 +00:00
|
|
|
}
|
|
|
|
}
|
2018-06-08 15:54:58 +00:00
|
|
|
return _response;
|
2018-04-18 14:38:06 +00:00
|
|
|
}
|
|
|
|
}
|