2015-12-25 22:42:14 +01:00
|
|
|
/*
|
|
|
|
This file is part of TALER
|
2016-01-26 17:21:17 +01:00
|
|
|
(C) 2015-2016 GNUnet e.V.
|
2015-12-25 22:42:14 +01:00
|
|
|
|
|
|
|
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, If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
import {amountToPretty, canonicalizeBaseUrl} from "../lib/wallet/helpers";
|
|
|
|
import {AmountJson, CreateReserveResponse} from "../lib/wallet/types";
|
|
|
|
|
2015-12-20 20:34:20 +01:00
|
|
|
"use strict";
|
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
declare var m: any;
|
2015-12-20 20:34:20 +01:00
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute something after a delay, with the possibility
|
|
|
|
* to reset the delay.
|
|
|
|
*/
|
|
|
|
class DelayTimer {
|
|
|
|
ms: number;
|
|
|
|
f;
|
|
|
|
timerId: number = null;
|
|
|
|
|
|
|
|
constructor(ms: number, f) {
|
|
|
|
this.f = f;
|
|
|
|
this.ms = ms;
|
|
|
|
}
|
|
|
|
|
|
|
|
bump() {
|
|
|
|
if (this.timerId !== null) {
|
|
|
|
window.clearTimeout(this.timerId);
|
|
|
|
}
|
|
|
|
const handler = () => {
|
|
|
|
this.f();
|
|
|
|
};
|
|
|
|
this.timerId = window.setTimeout(handler, this.ms);
|
2015-12-20 20:34:20 +01:00
|
|
|
}
|
2016-02-11 11:29:57 +01:00
|
|
|
}
|
2015-12-20 20:34:20 +01:00
|
|
|
|
2016-01-26 17:21:17 +01:00
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
class Controller {
|
|
|
|
url = null;
|
|
|
|
errorString = null;
|
|
|
|
isValidMint = false;
|
|
|
|
private timer: DelayTimer;
|
|
|
|
private request: XMLHttpRequest;
|
2015-12-20 20:34:20 +01:00
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
constructor() {
|
|
|
|
this.update();
|
|
|
|
this.timer = new DelayTimer(800, () => this.update());
|
|
|
|
}
|
|
|
|
|
|
|
|
update() {
|
|
|
|
const doUpdate = () => {
|
|
|
|
if (!this.url) {
|
|
|
|
this.errorString = i18n`Please enter a URL`;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.errorString = null;
|
|
|
|
let parsedUrl = URI(this.url);
|
|
|
|
if (parsedUrl.is("relative")) {
|
|
|
|
this.errorString = i18n`The URL you've entered is not valid (must be absolute)`;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const keysUrl = URI("/keys").absoluteTo(canonicalizeBaseUrl(this.url));
|
|
|
|
|
|
|
|
console.log(`requesting keys from '${keysUrl}'`);
|
|
|
|
|
|
|
|
this.request = new XMLHttpRequest();
|
|
|
|
this.request.onreadystatechange = () => {
|
|
|
|
if (this.request.readyState == XMLHttpRequest.DONE) {
|
|
|
|
switch (this.request.status) {
|
|
|
|
case 200:
|
|
|
|
this.isValidMint = true;
|
|
|
|
break;
|
|
|
|
case 0:
|
|
|
|
this.errorString = `unknown request error`;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
this.errorString = `request failed with status ${this.request.status}`;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
m.redraw();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
this.request.open("get", keysUrl.href());
|
|
|
|
this.request.send();
|
2016-02-09 21:56:06 +01:00
|
|
|
};
|
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
doUpdate();
|
|
|
|
m.redraw();
|
|
|
|
console.log("got update");
|
|
|
|
}
|
2015-12-20 20:34:20 +01:00
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
reset() {
|
|
|
|
this.isValidMint = false;
|
|
|
|
this.errorString = null;
|
|
|
|
if (this.request) {
|
|
|
|
this.request.abort();
|
|
|
|
this.request = null;
|
2016-02-09 21:56:06 +01:00
|
|
|
}
|
2016-02-11 11:29:57 +01:00
|
|
|
m.redraw();
|
|
|
|
}
|
2016-02-09 21:56:06 +01:00
|
|
|
|
2016-02-11 11:29:57 +01:00
|
|
|
confirmReserve(mint: string, amount: AmountJson, callback_url: string) {
|
|
|
|
const d = {mint, amount};
|
2016-02-09 21:56:06 +01:00
|
|
|
const cb = (rawResp) => {
|
|
|
|
if (!rawResp) {
|
|
|
|
throw Error("empty response");
|
|
|
|
}
|
|
|
|
if (!rawResp.error) {
|
|
|
|
const resp = CreateReserveResponse.checked(rawResp);
|
|
|
|
let q = {
|
|
|
|
mint: resp.mint,
|
|
|
|
reserve_pub: resp.reservePub,
|
2016-02-11 11:29:57 +01:00
|
|
|
amount_value: amount.value,
|
|
|
|
amount_fraction: amount.fraction,
|
|
|
|
amount_currency: amount.currency,
|
2016-02-09 21:56:06 +01:00
|
|
|
};
|
2016-02-11 11:29:57 +01:00
|
|
|
let url = URI(callback_url).addQuery(q);
|
2016-02-09 21:56:06 +01:00
|
|
|
if (!url.is("absolute")) {
|
|
|
|
throw Error("callback url is not absolute");
|
|
|
|
}
|
2016-02-11 11:29:57 +01:00
|
|
|
console.log("going to", url.href());
|
2016-02-09 21:56:06 +01:00
|
|
|
document.location.href = url.href();
|
2016-01-26 17:21:17 +01:00
|
|
|
} else {
|
2016-02-11 11:29:57 +01:00
|
|
|
this.reset();
|
|
|
|
this.errorString = (
|
|
|
|
`Oops, something went wrong.` +
|
|
|
|
`The wallet responded with error status (${rawResp.error}).`);
|
2016-01-26 17:21:17 +01:00
|
|
|
}
|
|
|
|
};
|
2016-02-09 21:56:06 +01:00
|
|
|
chrome.runtime.sendMessage({type: 'create-reserve', detail: d}, cb);
|
2016-02-11 11:29:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
onUrlChanged(url: string) {
|
|
|
|
this.reset();
|
|
|
|
this.url = url;
|
|
|
|
this.timer.bump();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export function main() {
|
|
|
|
const url = URI(document.location.href);
|
|
|
|
const query: any = URI.parseQuery(url.query());
|
|
|
|
const amount = AmountJson.checked(JSON.parse(query.amount));
|
|
|
|
const callback_url = query.callback_url;
|
|
|
|
|
|
|
|
var MintSelection = {
|
|
|
|
controller: () => new Controller(),
|
|
|
|
view(ctrl: Controller) {
|
|
|
|
let controls = [];
|
|
|
|
let mx = (...args) => controls.push(m(...args));
|
|
|
|
|
|
|
|
mx("p",
|
|
|
|
i18n`The bank wants to create a reserve over ${amountToPretty(
|
|
|
|
amount)}.`);
|
|
|
|
mx("input.url",
|
|
|
|
{
|
|
|
|
type: "text",
|
|
|
|
spellcheck: false,
|
|
|
|
oninput: m.withAttr("value", ctrl.onUrlChanged.bind(ctrl)),
|
|
|
|
});
|
|
|
|
|
|
|
|
if (ctrl.isValidMint) {
|
|
|
|
mx("button", {
|
|
|
|
onclick: () => ctrl.confirmReserve(ctrl.url,
|
|
|
|
amount,
|
|
|
|
callback_url)
|
|
|
|
},
|
|
|
|
"Confirm mint selection");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ctrl.errorString) {
|
|
|
|
mx("p", ctrl.errorString);
|
|
|
|
}
|
|
|
|
|
|
|
|
return m("div", controls);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
m.mount(document.getElementById("mint-selection"), MintSelection);
|
2015-12-20 20:34:20 +01:00
|
|
|
}
|