2016-01-26 17:21:17 +01:00
|
|
|
/*
|
|
|
|
This file is part of TALER
|
|
|
|
(C) 2016 GNUnet e.V.
|
|
|
|
|
|
|
|
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
|
2016-07-07 17:59:29 +02:00
|
|
|
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
2016-01-26 17:21:17 +01:00
|
|
|
*/
|
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
/**
|
2017-05-24 16:45:57 +02:00
|
|
|
* Small helper functions that don't fit anywhere else.
|
2016-02-11 11:29:57 +01:00
|
|
|
*/
|
|
|
|
|
2017-05-24 16:14:23 +02:00
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
2016-11-16 10:29:07 +01:00
|
|
|
import {AmountJson, Amounts} from "./types";
|
2017-04-20 03:09:25 +02:00
|
|
|
import URI = require("urijs");
|
2016-02-10 02:03:31 +01:00
|
|
|
|
2017-05-24 16:14:23 +02:00
|
|
|
/**
|
|
|
|
* Show an amount in a form suitable for the user.
|
|
|
|
* FIXME: In the future, this should consider currency-specific
|
|
|
|
* settings such as significant digits or currency symbols.
|
|
|
|
*/
|
2016-02-09 21:56:06 +01:00
|
|
|
export function amountToPretty(amount: AmountJson): string {
|
2017-05-28 01:10:54 +02:00
|
|
|
const x = amount.value + amount.fraction / Amounts.fractionalBase;
|
2016-02-09 21:56:06 +01:00
|
|
|
return `${x} ${amount.currency}`;
|
2016-02-11 11:29:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2016-03-01 19:39:17 +01:00
|
|
|
* Canonicalize a base url, typically for the exchange.
|
2016-02-11 11:29:57 +01:00
|
|
|
*
|
|
|
|
* See http://api.taler.net/wallet.html#general
|
|
|
|
*/
|
2016-09-12 20:25:56 +02:00
|
|
|
export function canonicalizeBaseUrl(url: string) {
|
2017-05-28 01:10:54 +02:00
|
|
|
const x = new URI(url);
|
2016-02-11 11:29:57 +01:00
|
|
|
if (!x.protocol()) {
|
|
|
|
x.protocol("https");
|
|
|
|
}
|
|
|
|
x.path(x.path() + "/").normalizePath();
|
2016-11-14 01:29:08 +01:00
|
|
|
x.fragment("");
|
2016-02-11 11:29:57 +01:00
|
|
|
x.query();
|
2017-05-28 01:10:54 +02:00
|
|
|
return x.href();
|
2016-02-11 11:29:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-11-13 08:16:12 +01:00
|
|
|
/**
|
|
|
|
* Convert object to JSON with canonical ordering of keys
|
|
|
|
* and whitespace omitted.
|
|
|
|
*/
|
|
|
|
export function canonicalJson(obj: any): string {
|
|
|
|
// Check for cycles, etc.
|
|
|
|
JSON.stringify(obj);
|
2017-05-28 01:10:54 +02:00
|
|
|
if (typeof obj === "string" || typeof obj === "number" || obj === null) {
|
|
|
|
return JSON.stringify(obj);
|
2016-11-13 08:16:12 +01:00
|
|
|
}
|
|
|
|
if (Array.isArray(obj)) {
|
2017-05-28 01:10:54 +02:00
|
|
|
const objs: string[] = obj.map((e) => canonicalJson(e));
|
|
|
|
return `[${objs.join(",")}]`;
|
2016-11-13 08:16:12 +01:00
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
const keys: string[] = [];
|
|
|
|
for (const key in obj) {
|
2016-11-13 08:16:12 +01:00
|
|
|
keys.push(key);
|
|
|
|
}
|
|
|
|
keys.sort();
|
|
|
|
let s = "{";
|
|
|
|
for (let i = 0; i < keys.length; i++) {
|
2017-05-28 01:10:54 +02:00
|
|
|
const key = keys[i];
|
2016-11-13 08:16:12 +01:00
|
|
|
s += JSON.stringify(key) + ":" + canonicalJson(obj[key]);
|
2017-05-28 01:10:54 +02:00
|
|
|
if (i !== keys.length - 1) {
|
2016-11-13 08:16:12 +01:00
|
|
|
s += ",";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return s + "}";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-28 16:27:34 +02:00
|
|
|
/**
|
|
|
|
* Check for deep equality of two objects.
|
|
|
|
* Only arrays, objects and primitives are supported.
|
|
|
|
*/
|
2016-11-13 08:16:12 +01:00
|
|
|
export function deepEquals(x: any, y: any): boolean {
|
|
|
|
if (x === y) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Array.isArray(x) && x.length !== y.length) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-05-28 01:10:54 +02:00
|
|
|
const p = Object.keys(x);
|
2016-11-13 08:16:12 +01:00
|
|
|
return Object.keys(y).every((i) => p.indexOf(i) !== -1) &&
|
|
|
|
p.every((i) => deepEquals(x[i], y[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-28 16:27:34 +02:00
|
|
|
/**
|
|
|
|
* Map from a collection to a list or results and then
|
|
|
|
* concatenate the results.
|
|
|
|
*/
|
2016-11-13 08:16:12 +01:00
|
|
|
export function flatMap<T, U>(xs: T[], f: (x: T) => U[]): U[] {
|
|
|
|
return xs.reduce((acc: U[], next: T) => [...f(next), ...acc], []);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-24 16:14:23 +02:00
|
|
|
/**
|
|
|
|
* Extract a numeric timstamp (in seconds) from the Taler date format
|
|
|
|
* ("/Date([n])/"). Returns null if input is not in the right format.
|
|
|
|
*/
|
2016-11-13 08:16:12 +01:00
|
|
|
export function getTalerStampSec(stamp: string): number | null {
|
|
|
|
const m = stamp.match(/\/?Date\(([0-9]*)\)\/?/);
|
|
|
|
if (!m) {
|
|
|
|
return null;
|
|
|
|
}
|
2017-05-28 01:10:54 +02:00
|
|
|
return parseInt(m[1], 10);
|
2016-11-13 08:16:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-24 16:14:23 +02:00
|
|
|
/**
|
|
|
|
* Get a JavaScript Date object from a Taler date string.
|
|
|
|
* Returns null if input is not in the right format.
|
|
|
|
*/
|
2016-11-13 08:16:12 +01:00
|
|
|
export function getTalerStampDate(stamp: string): Date | null {
|
2017-05-28 01:10:54 +02:00
|
|
|
const sec = getTalerStampSec(stamp);
|
2016-11-13 08:16:12 +01:00
|
|
|
if (sec == null) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return new Date(sec * 1000);
|
|
|
|
}
|
|
|
|
|