2023-01-10 17:31:01 +01:00
|
|
|
/*
|
|
|
|
This file is part of GNU Taler
|
|
|
|
(C) 2020 Taler Systems S.A.
|
|
|
|
|
|
|
|
GNU 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.
|
|
|
|
|
|
|
|
GNU 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
|
|
|
|
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Imports.
|
|
|
|
*/
|
2023-02-12 19:30:59 +01:00
|
|
|
import { Duration, j2s, NotificationType } from "@gnu-taler/taler-util";
|
|
|
|
import {
|
|
|
|
BankAccessApi,
|
|
|
|
BankApi,
|
|
|
|
NodeHttpLib,
|
|
|
|
WalletApiOperation,
|
|
|
|
} from "@gnu-taler/taler-wallet-core";
|
2023-01-10 17:31:01 +01:00
|
|
|
import { CoinConfig, defaultCoinConfig } from "../harness/denomStructures.js";
|
|
|
|
import {
|
|
|
|
BankService,
|
|
|
|
ExchangeService,
|
|
|
|
getPayto,
|
|
|
|
GlobalTestState,
|
|
|
|
MerchantService,
|
|
|
|
setupDb,
|
2023-02-12 19:30:59 +01:00
|
|
|
WalletClient,
|
|
|
|
WalletService,
|
2023-01-10 17:31:01 +01:00
|
|
|
} from "../harness/harness.js";
|
2023-02-12 19:30:59 +01:00
|
|
|
import { EnvOptions, SimpleTestEnvironmentNg } from "../harness/helpers.js";
|
|
|
|
import * as http from "node:http";
|
2023-01-10 17:31:01 +01:00
|
|
|
|
|
|
|
export async function createKycTestkudosEnvironment(
|
|
|
|
t: GlobalTestState,
|
|
|
|
coinConfig: CoinConfig[] = defaultCoinConfig.map((x) => x("TESTKUDOS")),
|
|
|
|
opts: EnvOptions = {},
|
2023-02-12 19:30:59 +01:00
|
|
|
): Promise<SimpleTestEnvironmentNg> {
|
2023-01-10 17:31:01 +01:00
|
|
|
const db = await setupDb(t);
|
|
|
|
|
|
|
|
const bank = await BankService.create(t, {
|
|
|
|
allowRegistrations: true,
|
|
|
|
currency: "TESTKUDOS",
|
|
|
|
database: db.connStr,
|
|
|
|
httpPort: 8082,
|
|
|
|
});
|
|
|
|
|
|
|
|
const exchange = ExchangeService.create(t, {
|
|
|
|
name: "testexchange-1",
|
|
|
|
currency: "TESTKUDOS",
|
|
|
|
httpPort: 8081,
|
|
|
|
database: db.connStr,
|
|
|
|
});
|
|
|
|
|
|
|
|
const merchant = await MerchantService.create(t, {
|
|
|
|
name: "testmerchant-1",
|
|
|
|
currency: "TESTKUDOS",
|
|
|
|
httpPort: 8083,
|
|
|
|
database: db.connStr,
|
|
|
|
});
|
|
|
|
|
|
|
|
const exchangeBankAccount = await bank.createExchangeAccount(
|
|
|
|
"myexchange",
|
|
|
|
"x",
|
|
|
|
);
|
|
|
|
exchange.addBankAccount("1", exchangeBankAccount);
|
|
|
|
|
|
|
|
bank.setSuggestedExchange(exchange, exchangeBankAccount.accountPaytoUri);
|
|
|
|
|
|
|
|
await bank.start();
|
|
|
|
|
|
|
|
await bank.pingUntilAvailable();
|
|
|
|
|
|
|
|
const ageMaskSpec = opts.ageMaskSpec;
|
|
|
|
|
|
|
|
if (ageMaskSpec) {
|
|
|
|
exchange.enableAgeRestrictions(ageMaskSpec);
|
|
|
|
// Enable age restriction for all coins.
|
|
|
|
exchange.addCoinConfigList(
|
|
|
|
coinConfig.map((x) => ({
|
|
|
|
...x,
|
|
|
|
name: `${x.name}-age`,
|
|
|
|
ageRestricted: true,
|
|
|
|
})),
|
|
|
|
);
|
|
|
|
// For mixed age restrictions, we also offer coins without age restrictions
|
|
|
|
if (opts.mixedAgeRestriction) {
|
|
|
|
exchange.addCoinConfigList(
|
|
|
|
coinConfig.map((x) => ({ ...x, ageRestricted: false })),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
exchange.addCoinConfigList(coinConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
await exchange.modifyConfig(async (config) => {
|
|
|
|
const myprov = "kyc-provider-myprov";
|
|
|
|
config.setString(myprov, "cost", "0");
|
|
|
|
config.setString(myprov, "logic", "oauth2");
|
|
|
|
config.setString(myprov, "provided_checks", "dummy1");
|
|
|
|
config.setString(myprov, "user_type", "individual");
|
|
|
|
config.setString(myprov, "kyc_oauth2_validity", "forever");
|
|
|
|
config.setString(
|
|
|
|
myprov,
|
|
|
|
"kyc_oauth2_auth_url",
|
|
|
|
"http://localhost:6666/oauth/v2/token",
|
|
|
|
);
|
|
|
|
config.setString(
|
|
|
|
myprov,
|
|
|
|
"kyc_oauth2_login_url",
|
|
|
|
"http://localhost:6666/oauth/v2/login",
|
|
|
|
);
|
|
|
|
config.setString(
|
|
|
|
myprov,
|
|
|
|
"kyc_oauth2_info_url",
|
2023-02-12 19:30:59 +01:00
|
|
|
"http://localhost:6666/oauth/v2/info",
|
2023-01-10 17:31:01 +01:00
|
|
|
);
|
2023-01-19 20:16:42 +01:00
|
|
|
config.setString(myprov, "kyc_oauth2_client_id", "taler-exchange");
|
|
|
|
config.setString(myprov, "kyc_oauth2_client_secret", "exchange-secret");
|
2023-02-12 19:30:59 +01:00
|
|
|
config.setString(myprov, "kyc_oauth2_post_url", "https://taler.net");
|
2023-01-19 20:16:42 +01:00
|
|
|
|
2023-01-10 17:31:01 +01:00
|
|
|
config.setString(
|
2023-01-19 20:16:42 +01:00
|
|
|
"kyc-legitimization-withdraw1",
|
|
|
|
"operation_type",
|
|
|
|
"withdraw",
|
2023-01-10 17:31:01 +01:00
|
|
|
);
|
|
|
|
config.setString(
|
2023-01-19 20:16:42 +01:00
|
|
|
"kyc-legitimization-withdraw1",
|
|
|
|
"required_checks",
|
|
|
|
"dummy1",
|
2023-01-10 17:31:01 +01:00
|
|
|
);
|
2023-01-19 20:16:42 +01:00
|
|
|
config.setString("kyc-legitimization-withdraw1", "timeframe", "1d");
|
2023-01-10 17:31:01 +01:00
|
|
|
config.setString(
|
2023-01-19 20:16:42 +01:00
|
|
|
"kyc-legitimization-withdraw1",
|
|
|
|
"threshold",
|
|
|
|
"TESTKUDOS:5",
|
2023-01-10 17:31:01 +01:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
await exchange.start();
|
|
|
|
await exchange.pingUntilAvailable();
|
|
|
|
|
|
|
|
merchant.addExchange(exchange);
|
|
|
|
|
|
|
|
await merchant.start();
|
|
|
|
await merchant.pingUntilAvailable();
|
|
|
|
|
|
|
|
await merchant.addInstance({
|
|
|
|
id: "default",
|
|
|
|
name: "Default Instance",
|
|
|
|
paytoUris: [getPayto("merchant-default")],
|
|
|
|
defaultWireTransferDelay: Duration.toTalerProtocolDuration(
|
|
|
|
Duration.fromSpec({ minutes: 1 }),
|
|
|
|
),
|
|
|
|
});
|
|
|
|
|
|
|
|
await merchant.addInstance({
|
|
|
|
id: "minst1",
|
|
|
|
name: "minst1",
|
|
|
|
paytoUris: [getPayto("minst1")],
|
|
|
|
defaultWireTransferDelay: Duration.toTalerProtocolDuration(
|
|
|
|
Duration.fromSpec({ minutes: 1 }),
|
|
|
|
),
|
|
|
|
});
|
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
const walletService = new WalletService(t, {
|
|
|
|
name: "wallet",
|
|
|
|
useInMemoryDb: true,
|
|
|
|
});
|
|
|
|
await walletService.start();
|
|
|
|
await walletService.pingUntilAvailable();
|
2023-01-10 17:31:01 +01:00
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
const walletClient = new WalletClient({
|
|
|
|
unixPath: walletService.socketPath,
|
|
|
|
onNotification(n) {
|
|
|
|
console.log("got notification", n);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
await walletClient.connect();
|
|
|
|
await walletClient.client.call(WalletApiOperation.InitWallet, {
|
|
|
|
skipDefaults: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
console.log("setup done!");
|
2023-01-10 17:31:01 +01:00
|
|
|
|
|
|
|
return {
|
|
|
|
commonDb: db,
|
|
|
|
exchange,
|
|
|
|
merchant,
|
2023-02-12 19:30:59 +01:00
|
|
|
walletClient,
|
|
|
|
walletService,
|
2023-01-10 17:31:01 +01:00
|
|
|
bank,
|
|
|
|
exchangeBankAccount,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
interface TestfakeKycService {
|
|
|
|
stop: () => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
function splitInTwoAt(s: string, separator: string): [string, string] {
|
|
|
|
const idx = s.indexOf(separator);
|
|
|
|
if (idx === -1) {
|
|
|
|
return [s, ""];
|
|
|
|
}
|
|
|
|
return [s.slice(0, idx), s.slice(idx + 1)];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Testfake for the kyc service that the exchange talks to.
|
|
|
|
*/
|
|
|
|
async function runTestfakeKycService(): Promise<TestfakeKycService> {
|
|
|
|
const server = http.createServer((req, res) => {
|
|
|
|
const requestUrl = req.url!;
|
|
|
|
console.log(`kyc: got ${req.method} request`, requestUrl);
|
|
|
|
|
|
|
|
const [path, query] = splitInTwoAt(requestUrl, "?");
|
|
|
|
|
|
|
|
const qp = new URLSearchParams(query);
|
|
|
|
|
|
|
|
if (path === "/oauth/v2/login") {
|
|
|
|
// Usually this would render some HTML page for the user to log in,
|
|
|
|
// but we return JSON here.
|
|
|
|
const redirUri = new URL(qp.get("redirect_uri")!);
|
|
|
|
redirUri.searchParams.set("code", "code_is_ok");
|
|
|
|
res.writeHead(200, { "Content-Type": "application/json" });
|
|
|
|
res.end(
|
|
|
|
JSON.stringify({
|
|
|
|
redirect_uri: redirUri.href,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
} else if (path === "/oauth/v2/token") {
|
|
|
|
let reqBody = "";
|
|
|
|
req.on("data", (x) => {
|
|
|
|
reqBody += x;
|
|
|
|
});
|
|
|
|
|
|
|
|
req.on("end", () => {
|
|
|
|
console.log("login request body:", reqBody);
|
|
|
|
|
|
|
|
res.writeHead(200, { "Content-Type": "application/json" });
|
|
|
|
// Normally, the access_token would also include which user we're trying
|
|
|
|
// to get info about, but we (for now) skip it in this test.
|
|
|
|
res.end(
|
|
|
|
JSON.stringify({
|
|
|
|
access_token: "exchange_access_token",
|
|
|
|
token_type: "Bearer",
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
} else if (path === "/oauth/v2/info") {
|
|
|
|
console.log("authorization header:", req.headers.authorization);
|
|
|
|
res.writeHead(200, { "Content-Type": "application/json" });
|
|
|
|
res.end(
|
|
|
|
JSON.stringify({
|
|
|
|
status: "success",
|
|
|
|
data: {
|
|
|
|
id: "foobar",
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
res.writeHead(400, { "Content-Type": "application/json" });
|
|
|
|
res.end(JSON.stringify({ code: 1, message: "bad request" }));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
await new Promise<void>((resolve, reject) => {
|
|
|
|
server.listen(6666, () => resolve());
|
|
|
|
});
|
|
|
|
return {
|
|
|
|
stop() {
|
|
|
|
server.close();
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-01-10 17:31:01 +01:00
|
|
|
export async function runKycTest(t: GlobalTestState) {
|
|
|
|
// Set up test environment
|
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
const { walletClient, bank, exchange, merchant } =
|
2023-01-10 17:31:01 +01:00
|
|
|
await createKycTestkudosEnvironment(t);
|
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
const kycServer = await runTestfakeKycService();
|
|
|
|
|
2023-01-10 17:31:01 +01:00
|
|
|
// Withdraw digital cash into the wallet.
|
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
const amount = "TESTKUDOS:20";
|
|
|
|
const user = await BankApi.createRandomBankUser(bank);
|
|
|
|
const wop = await BankAccessApi.createWithdrawalOperation(bank, user, amount);
|
2023-01-10 17:31:01 +01:00
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
// Hand it to the wallet
|
|
|
|
|
|
|
|
await walletClient.client.call(
|
|
|
|
WalletApiOperation.GetWithdrawalDetailsForUri,
|
|
|
|
{
|
|
|
|
talerWithdrawUri: wop.taler_withdraw_uri,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
// Withdraw
|
|
|
|
|
|
|
|
const kycNotificationCond = walletClient.waitForNotificationCond((x) => {
|
|
|
|
if (x.type === NotificationType.WithdrawalKycRequested) {
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
const withdrawalDoneCond = walletClient.waitForNotificationCond(
|
|
|
|
(x) => x.type === NotificationType.WithdrawGroupFinished,
|
|
|
|
);
|
|
|
|
|
|
|
|
await walletClient.client.call(
|
|
|
|
WalletApiOperation.AcceptBankIntegratedWithdrawal,
|
|
|
|
{
|
|
|
|
exchangeBaseUrl: exchange.baseUrl,
|
|
|
|
talerWithdrawUri: wop.taler_withdraw_uri,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
// Confirm it
|
|
|
|
|
|
|
|
await BankApi.confirmWithdrawalOperation(bank, user, wop);
|
|
|
|
|
|
|
|
const kycNotif = await kycNotificationCond;
|
|
|
|
|
|
|
|
console.log("got kyc notification:", j2s(kycNotif));
|
|
|
|
|
|
|
|
// We now simulate the user interacting with the KYC service,
|
|
|
|
// which would usually done in the browser.
|
|
|
|
|
|
|
|
const httpClient = new NodeHttpLib();
|
|
|
|
const kycServerResp = await httpClient.get(kycNotif.kycUrl);
|
|
|
|
const kycLoginResp = await kycServerResp.json();
|
|
|
|
console.log("kyc server resp:", j2s(kycLoginResp));
|
|
|
|
const kycProofUrl = kycLoginResp.redirect_uri;
|
|
|
|
const proofHttpResp = await httpClient.get(kycProofUrl);
|
|
|
|
console.log("proof resp status", proofHttpResp.status);
|
|
|
|
console.log("resp headers", proofHttpResp.headers.toJSON());
|
|
|
|
|
|
|
|
// Now that KYC is done, withdrawal should finally succeed.
|
|
|
|
|
|
|
|
await withdrawalDoneCond;
|
2023-01-10 17:31:01 +01:00
|
|
|
|
2023-02-12 19:30:59 +01:00
|
|
|
kycServer.stop();
|
2023-01-10 17:31:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
runKycTest.suites = ["wallet"];
|