/*
This file is part of TALER
Copyright (C) 2014, 2015 Christian Grothoff (and other contributing authors)
TALER is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
TALER 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 General Public License for more details.
You should have received a copy of the GNU General Public License along with
TALER; see the file COPYING. If not, see
*/
"use strict";
declare var Module : any;
// Size of a native pointer.
const PTR_SIZE = 4;
let getEmsc = Module.cwrap;
var emsc = {
free: (ptr) => Module._free(ptr),
get_value: getEmsc('TALER_WR_get_value',
'number',
['number']),
get_fraction: getEmsc('TALER_WR_get_fraction',
'number',
['number']),
get_currency: getEmsc('TALER_WR_get_currency',
'string',
['number']),
amount_add: getEmsc('TALER_amount_add',
'void',
['number', 'number', 'number']),
amount_subtract: getEmsc('TALER_amount_subtract',
'void',
['number', 'number', 'number']),
amount_normalize: getEmsc('TALER_amount_normalize',
'void',
['number']),
amount_cmp: getEmsc('TALER_amount_cmp',
'number',
['number', 'number'])
};
var emscAlloc = {
get_amount: getEmsc('TALER_WRALL_get_amount',
'number',
['number', 'number', 'number', 'string']),
eddsa_key_create: getEmsc('GNUNET_CRYPTO_eddsa_key_create',
'number'),
eddsa_public_key_from_private: getEmsc('TALER_WRALL_eddsa_public_key_from_private',
'number',
['number']),
data_to_string_alloc: getEmsc('GNUNET_STRINGS_data_to_string_alloc',
'number',
['number', 'number']),
malloc: (size : number) => Module._malloc(size),
};
abstract class ArenaObject {
nativePtr: number;
arena: Arena;
abstract destroy(): void;
constructor(arena?: Arena) {
this.nativePtr = 0;
if (!arena)
arena = defaultArena;
arena.put(this);
this.arena = arena;
}
}
class Arena {
heap: Array;
constructor () {
this.heap = [];
}
put(obj) {
this.heap.push(obj);
}
destroy(obj) {
// XXX: todo
}
}
// Arena to track allocations that do not use an explicit arena.
var defaultArena = new Arena();
class Amount extends ArenaObject {
constructor(args?: any, arena?: Arena) {
super(arena);
if (args) {
this.nativePtr = emscAlloc.get_amount(args.value, 0, args.fraction, args.currency);
} else {
this.nativePtr = emscAlloc.get_amount(0, 0, 0, "");
}
}
destroy() {
if (this.nativePtr != 0) {
emsc.free(this.nativePtr);
}
}
get value() {
return emsc.get_value(this.nativePtr);
}
get fraction() {
return emsc.get_fraction(this.nativePtr);
}
get currency() {
return emsc.get_currency(this.nativePtr);
}
toJson() {
return {
value: emsc.get_value(this.nativePtr),
fraction: emsc.get_fraction(this.nativePtr),
currency: emsc.get_currency(this.nativePtr)
};
}
/**
* Add an amount to this amount.
*/
add(a) {
let res = emsc.amount_add(this.nativePtr, a.nativePtr, this.nativePtr);
if (res < 1) {
// Overflow
return false;
}
return true;
}
/**
* Perform saturating subtraction on amounts.
*/
sub(a) {
let res = emsc.amount_subtract(this.nativePtr, a.nativePtr, this.nativePtr);
if (res == 0) {
// Underflow
return false;
}
if (res > 0) {
return true;
}
throw "Incompatible currencies";
}
cmp(a) {
return emsc.amount_cmp(this.nativePtr, a.nativePtr);
}
normalize() {
emsc.amount_normalize(this.nativePtr);
}
}
abstract class PackedArenaObject extends ArenaObject {
size: number;
encode(): string {
var d = emscAlloc.data_to_string_alloc(this.nativePtr, this.size);
var s = Module.Pointer_stringify(d);
emsc.free(d);
return s;
}
}
class EddsaPrivateKey extends PackedArenaObject {
static create(a?: Arena): EddsaPrivateKey {
let k = new EddsaPrivateKey(a);
k.nativePtr = emscAlloc.eddsa_key_create();
return k;
}
get size() { return 32; }
getPublicKey(): EddsaPublicKey {
let pk = new EddsaPublicKey(this.arena);
pk.nativePtr = emscAlloc.eddsa_public_key_from_private(this.nativePtr);
return pk;
}
}
class EddsaPublicKey extends PackedArenaObject {
destroy() {
// TODO
}
get size() {
return 32;
}
}
class RsaBlindingKey extends ArenaObject {
destroy() {
// TODO
}
}
class HashCode extends ArenaObject {
destroy() {
// TODO
}
}
class ByteArray extends PackedArenaObject {
destroy() {
// TODO
}
}
class RsaPublicKey extends ArenaObject {
destroy() {
// TODO
}
}
function rsaBlind(hashCode: HashCode,
blindingKey: RsaBlindingKey,
pkey: RsaPublicKey,
arena?: Arena): ByteArray
{
return null;
}