По сути, я создал собственный расчетный рельс для corda-отладчика , и теперь я пытаюсь создать пользовательский интерфейс для демонстрации.У меня возникла проблема при попытке создать обязательство с помощью вызова API.Я использую встроенный jettywebserver и закодировал свой собственный API, используя javascript для отправки запроса.Надеюсь получить совет / понимание того, где я могу поступать неправильно, я относительно новичок в этом и не уверен, что приведенные ниже коды помогут вам всем лучше проконсультировать меня по этому вопросу, но, пожалуйста, дайте мне знать, если больше информациидолжен быть предоставлен.Заранее спасибо!
Это пользовательский интерфейс, который я создал (извините, недостаточно репутации для публикации изображения напрямую):
https://imgur.com/xPUQdUj
и после нажатияКнопка «Создать обязательство» возвращает:
https://imgur.com/3X1FPp6
Мне кажется, что я не получаю / не вызываю API должным образом, но я просто не могу понять, что пошло не так, потому чтокоды мне подходят.
Это мой index.html
файл:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Settler CorDapp</title>
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css"
integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap-theme.min.css"
integrity="sha384-rHyoN1iRsVXV4nD0JutlnGaslCJuC7uwjduW9SVrLvRYooPp2bWYgmgJQIXwl/Sp" crossorigin="anonymous">
<link rel="stylesheet" type="text/css" href="css/index.css">
<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.1.1/jquery.min.js"></script>
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js"
integrity="sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa"
crossorigin="anonymous"></script>
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.6.0-rc.1/angular.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/angular-ui-bootstrap/2.2.0/ui-bootstrap-tpls.min.js"></script>
<script src="js/main.js"></script>
<script src="js/createObligationModal.js"></script>
<script src="js/settleModal.js"></script>
</head>
<body ng-app="demoAppModule" ng-controller="DemoAppCtrl as demoApp">
<nav class="navbar navbar-default">
<div class="container-fluid">
<div class="navbar-header">
<a class="navbar-brand" href="#">{{demoApp.thisNode}}</a>
</div>
<button ng-click="demoApp.openCreateObligationModal()" type="button" class="btn btn-primary navbar-btn">Create Obligation</button>
<button ng-click="demoApp.refresh()" type="button" class="btn btn-default navbar-btn"><span
class="glyphicon glyphicon-refresh"></span></button>
</div>
</nav>
<script type="text/ng-template" id="createObligationModal.html">
<div class="modal-header">
<h4 class="modal-title">Add new Obligation</h4>
</div>
<form>
<div class="modal-body">
<div class="form-group">
<label for="createObligationCounterparty" class="control-label">Counter-party:</label>
<select ng-model="createObligationModal.form.counterparty" class="form-control" id="createObligationCounterparty"
ng-options="peer as peer for peer in createObligationModal.peers">
</select>
</div>
<div class="form-group">
<label for="createObligationCurrency" class="control-label">Currency (ISO code):</label>
<input type="text" ng-model="createObligationModal.form.currency" class="form-control" id="createObligationCurrency">
</div>
<div class="form-group">
<label for="createObligationAmount" class="control-label">Amount (Int):</label>
<input type="text" ng-model="createObligationModal.form.amount" class="form-control" id="createObligationAmount">
</div>
<div class="form-group">
<label for="createObligationDueDate" class="control-label">Due Date:</label>
<input type="date" ng-model="createObligationModal.form.duedate" class="form-control" id="createObligationDueDate" min="2020-01-01" max="2022-12-31">
</div>
<div ng-show="createObligationModal.formError" class="form-group">
<div class="alert alert-danger" role="alert">
<span class="glyphicon glyphicon-exclamation-sign" aria-hidden="true"></span>
<span class="sr-only">Error:</span>
Enter valid Obligation creation parameters
</div>
</div>
</div>
<div class="modal-footer">
<button ng-click="createObligationModal.cancel()" type="button" class="btn btn-default">Close</button>
<button ng-click="createObligationModal.create()" type="button" class="btn btn-primary">Create Obligation</button>
</div>
</form>
</script>
<script type="text/ng-template" id="createObligationMsgModal.html">
<div class="modal-body" id="create-Obligation-modal-body">
{{ createObligationMsgModal.message }}
</div>
</script>
<script type="text/ng-template" id="settleModal.html">
<div class="modal-header">
<h4 class="modal-title">Settle Obligation</h4>
</div>
<form>
<div class="modal-body">
<div class="form-group">
<label for="settleCurrency" class="control-label">Currency (ISO code):</label>
<input type="text" ng-model="settleModal.form.currency" class="form-control" id="settleCurrency">
</div>
<div class="form-group">
<label for="settleAmount" class="control-label">Amount (Int):</label>
<input type="text" ng-model="settleModal.form.amount" class="form-control" id="settleAmount">
</div>
<div ng-show="settleModal.formError" class="form-group">
<div class="alert alert-danger" role="alert">
<span class="glyphicon glyphicon-exclamation-sign" aria-hidden="true"></span>
<span class="sr-only">Error:</span>
Enter valid Obligation settle parameters.
</div>
</div>
</div>
<div class="modal-footer">
<button ng-click="settleModal.cancel()" type="button" class="btn btn-default">Close</button>
<button ng-click="settleModal.settle()" type="button" class="btn btn-primary">Settle</button>
</div>
</form>
</script>
<script type="text/ng-template" id="settleMsgModal.html">
<div class="modal-body" id="settle-modal-body">
{{ settleMsgModal.message }}
</div>
</script>
<div class="row">
<div class="col-md-1"></div>
<div class="col-md-10">
<div ng-show="!demoApp.obligations.length" class="panel panel-primary">
<div class="panel-heading">
<h3 class="panel-title">There are no recorded Obligations</h3>
</div>
<div class="panel-body">Use the "Create Obligation" button to send an Obligation to a peer.</div>
</div>
<div ng-show="demoApp.obligations.length" class="panel panel-primary">
<div class="panel-heading">
<h3 class="panel-title">Recorded Obligations:</h3>
</div>
<div class="panel-body">
<table class="table">
<thead>
<tr>
<th>From</th>
<th>To</th>
<th>Amount</th>
<th>Paid</th>
<th>Actions</th>
</tr>
</thead>
<tbody>
<tr ng-repeat="obligation in demoApp.obligations">
<td class="vert-align">{{obligation.obligee.substring(0,30)}}</td>
<td class="vert-align">{{obligation.obligor.substring(0,30)}}</td>
<td class="vert-align">{{obligation.faceAmount}}</td>
<td class="vert-align">{{obligation.amountPaid}}</td>
<td>
<div class="btn-group" role="group">
<button ng-click="demoApp.openSettleModal(obligation.linearId.id)" type="button" class="btn btn-primary">Settle
</button>
</div>
</td>
</tr>
</tbody>
</table>
</div>
</div>
<div class="col-md-1"></div>
</div>
</div>
</body>
</html>
Вот мои ObligationPlugin.kt
и ObligationApi.kt
файлы соответственно: package com.r3.corda.finance.obligation
import net.corda.core.messaging.CordaRPCOps
import net.corda.webserver.services.WebServerPluginRegistry
import java.util.function.Function
class ObligationPlugin : WebServerPluginRegistry {
override val webApis: List<Function<CordaRPCOps, out Any>> = listOf(Function(::ObligationApi))
override val staticServeDirs: Map<String, String> = mapOf(
"obligation" to javaClass.classLoader.getResource("settlerWeb").toExternalForm()
)
}
package com.r3.corda.finance.obligation
import com.r3.corda.finance.obligation.flows.CreateObligation
import com.r3.corda.finance.obligation.states.Obligation
import net.corda.core.contracts.Amount
import net.corda.core.messaging.CordaRPCOps
import net.corda.core.utilities.getOrThrow
import java.util.*
import javax.ws.rs.GET
import javax.ws.rs.Path
import javax.ws.rs.Produces
import javax.ws.rs.QueryParam
import javax.ws.rs.core.MediaType
import javax.ws.rs.core.Response
@Path("obligation")
class ObligationApi(val rpcOps: CordaRPCOps) {
private val myIdentity = rpcOps.nodeInfo().legalIdentities.first()
@GET
@Path("me")
@Produces(MediaType.APPLICATION_JSON)
fun me() = mapOf("me" to myIdentity)
@GET
@Path("peers")
@Produces(MediaType.APPLICATION_JSON)
fun peers() = mapOf("peers" to rpcOps.networkMapSnapshot()
.filter { nodeInfo -> nodeInfo.legalIdentities.first() != myIdentity }
.map { it.legalIdentities.first().name.organisation })
@GET
@Path("issue-obligation")
fun issueObligation(@QueryParam(value = "party") party: String,
@QueryParam(value = "currency") currency: String,
@QueryParam(value = "amount") amount: Int,
@QueryParam(value = "duedate") duedate: Date
): Response {
// 1. Get party objects for the counterparty.
val obligorIdentity = rpcOps.partiesFromName(party, exactMatch = false).singleOrNull()
?: throw IllegalStateException("Couldn't lookup node identity for $party.")
// 2. Create an amount object.
val issueAmount = Amount(amount.toLong() * 100, Currency.getInstance(currency))
// 3. Convert duedate to unix timestamp (in seconds)
val unixTime = (duedate.time) / 1000
// 4. Retrieve obligee identity
// val obligeeIdentity = myIdentity
// 5. Start the IssueObligation flow. We block and wait for the flow to return.
val (status, message) = try {
val flowHandle = rpcOps.startFlowDynamic(
CreateObligation.Initiator::class.java,
issueAmount,
CreateObligation.InitiatorRole.OBLIGEE,
obligorIdentity,
unixTime,
true
)
val result = flowHandle.returnValue.getOrThrow()
flowHandle.close()
Response.Status.CREATED to "Transaction id ${result.id} committed to ledger.\n${result.outputs}"
} catch (e: Exception) {
Response.Status.BAD_REQUEST to e.message
}
// 4. Return the result.
return Response.status(status).entity(message).build()
}
@GET
@Path("obligations")
@Produces(MediaType.APPLICATION_JSON)
fun obligations(): List<Obligation<*>> {
val statesAndRefs = rpcOps.vaultQuery(Obligation::class.java).states
return statesAndRefs
.map { stateAndRef -> stateAndRef.state.data }
.map { state ->
// We map the anonymous lender and borrower to well-known identities if possible.
val possiblyWellKnownLender = rpcOps.wellKnownPartyFromAnonymous(state.obligee) ?: state.obligee
val possiblyWellKnownBorrower = rpcOps.wellKnownPartyFromAnonymous(state.obligor) ?: state.obligor
Obligation(state.faceAmount,
possiblyWellKnownBorrower,
possiblyWellKnownLender,
state.dueBy,
state.createdAt,
state.settlementMethod,
state.payments,
state.linearId)
}
}
}
и следующие файлы main.js
и createObligationModal.js
соответственно:
"use strict";
// Define your backend here.
angular.module('demoAppModule', ['ui.bootstrap']).controller('DemoAppCtrl', function ($http, $location, $uibModal) {
const demoApp = this;
const apiBaseURL = "/api/obligation/";
// Retrieves the identity of this and other nodes.
let peers = [];
$http.get(apiBaseURL + "me").then((response) => demoApp.thisNode = response.data.me);
$http.get(apiBaseURL + "peers").then((response) => peers = response.data.peers);
/** Displays the obligation creation modal. */
demoApp.openCreateObligationModal = () => {
const createObligationModal = $uibModal.open({
templateUrl: 'createObligationModal.html',
controller: 'CreateObligationModalCtrl',
controllerAs: 'createObligationModal',
resolve: {
demoApp: () => demoApp,
apiBaseURL: () => apiBaseURL,
peers: () => peers,
refreshCallback: () => demoApp.refresh
}
});
// Ignores the modal result events.
createObligationModal.result.then(() => {
}, () => {
});
};
/** Displays the Obligation settlement modal. */
demoApp.openSettleModal = (id) => {
const settleModal = $uibModal.open({
templateUrl: 'settleModal.html',
controller: 'SettleModalCtrl',
controllerAs: 'settleModal',
resolve: {
demoApp: () => demoApp,
apiBaseURL: () => apiBaseURL,
id: () => id,
refreshCallback: () => demoApp.refresh
}
});
settleModal.result.then(() => {
}, () => {
});
};
/** Refreshes the front-end. */
demoApp.refresh = () => {
// Update the list of Obligations.
$http.get(apiBaseURL + "obligations").then((response) => demoApp.obligations =
Object.keys(response.data).map((key) => response.data[key]));
}
demoApp.refresh();
});
// Causes the webapp to ignore unhandled modal dismissals.
angular.module('demoAppModule').config(['$qProvider', function ($qProvider) {
$qProvider.errorOnUnhandledRejections(false);
}]);
"use strict";
angular.module('demoAppModule').controller('CreateObligationModalCtrl', function ($http, $uibModalInstance, $uibModal, apiBaseURL, peers, refreshCallback) {
const createObligationModal = this;
createObligationModal.peers = peers;
createObligationModal.form = {};
createObligationModal.formError = false;
/** Validate and create an Obligation. */
createObligationModal.create = () => {
if (invalidFormInput()) {
createObligationModal.formError = true;
} else {
createObligationModal.formError = false;
const party = createObligationModal.form.counterparty;
const currency = createObligationModal.form.currency;
const amount = createObligationModal.form.amount;
const duedate = createObligationModal.form.duedate;
$uibModalInstance.close();
// We define the Obligation creation endpoint.
const issueObligationEndpoint =
apiBaseURL +
`issue-obligation?party=${party}¤cy=${currency}&amount=${amount}&duedate=${duedate}`;
// We hit the endpoint to create the Obligation and handle success/failure responses.
$http.get(issueObligationEndpoint).then(
(result) => {
createObligationModal.displayMessage(result);
refreshCallback();
},
(result) => {
createObligationModal.displayMessage(result);
refreshCallback();
}
);
}
};
/** Displays the success/failure response from attempting to create an Obligation. */
createObligationModal.displayMessage = (message) => {
const createObligationMsgModal = $uibModal.open({
templateUrl: 'createObligationMsgModal.html',
controller: 'createObligationMsgModalCtrl',
controllerAs: 'createObligationMsgModal',
resolve: {
message: () => message
}
});
// No behaviour on close / dismiss.
createObligationMsgModal.result.then(() => {}, () => {});
};
/** Closes the Obligation creation modal. */
createObligationModal.cancel = () => $uibModalInstance.dismiss();
// Validates the Obligation.
function invalidFormInput() {
return isNaN(createObligationModal.form.amount) || (createObligationModal.form.counterparty === undefined);
}
});
// Controller for the success/fail modal.
angular.module('demoAppModule').controller('createObligationMsgModalCtrl', function ($uibModalInstance, message) {
const createObligationMsgModal = this;
createObligationMsgModal.message = message.data;
});
продолжить ...