mirror of
https://github.com/casdoor/casdoor.git
synced 2025-05-22 18:25:47 +08:00

* feat: support alipay payment provider * feat: update notify params * feat: update root cert * feat: update ProviderEditPage.js * feat: gofumpt
239 lines
8.0 KiB
JavaScript
239 lines
8.0 KiB
JavaScript
// Copyright 2022 The Casdoor Authors. All Rights Reserved.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
import React from "react";
|
|
import {Button, Result, Spin} from "antd";
|
|
import * as PaymentBackend from "./backend/PaymentBackend";
|
|
import * as PricingBackend from "./backend/PricingBackend";
|
|
import * as SubscriptionBackend from "./backend/SubscriptionBackend";
|
|
import * as Setting from "./Setting";
|
|
import i18next from "i18next";
|
|
|
|
class PaymentResultPage extends React.Component {
|
|
constructor(props) {
|
|
super(props);
|
|
const params = new URLSearchParams(window.location.search);
|
|
this.state = {
|
|
classes: props,
|
|
owner: props.match?.params?.organizationName ?? props.match?.params?.owner ?? null,
|
|
paymentName: props.match?.params?.paymentName ?? null,
|
|
pricingName: props.pricingName ?? props.match?.params?.pricingName ?? null,
|
|
subscriptionName: params.get("subscription"),
|
|
payment: null,
|
|
pricing: props.pricing ?? null,
|
|
subscription: props.subscription ?? null,
|
|
timeout: null,
|
|
};
|
|
}
|
|
|
|
UNSAFE_componentWillMount() {
|
|
this.getPayment();
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
if (this.state.timeout !== null) {
|
|
clearTimeout(this.state.timeout);
|
|
}
|
|
}
|
|
|
|
setStateAsync(state) {
|
|
return new Promise((resolve, reject) => {
|
|
this.setState(state, () => {
|
|
resolve();
|
|
});
|
|
});
|
|
}
|
|
|
|
onUpdatePricing(pricing) {
|
|
this.props.onUpdatePricing(pricing);
|
|
}
|
|
|
|
async getPayment() {
|
|
if (!(this.state.owner && (this.state.paymentName || (this.state.pricingName && this.state.subscriptionName)))) {
|
|
return ;
|
|
}
|
|
try {
|
|
// loading price & subscription
|
|
if (this.state.pricingName && this.state.subscriptionName) {
|
|
if (!this.state.pricing) {
|
|
const res = await PricingBackend.getPricing(this.state.owner, this.state.pricingName);
|
|
if (res.status !== "ok") {
|
|
throw new Error(res.msg);
|
|
}
|
|
const pricing = res.data;
|
|
await this.setStateAsync({
|
|
pricing: pricing,
|
|
});
|
|
}
|
|
if (!this.state.subscription) {
|
|
const res = await SubscriptionBackend.getSubscription(this.state.owner, this.state.subscriptionName);
|
|
if (res.status !== "ok") {
|
|
throw new Error(res.msg);
|
|
}
|
|
const subscription = res.data;
|
|
await this.setStateAsync({
|
|
subscription: subscription,
|
|
});
|
|
}
|
|
const paymentName = this.state.subscription.payment;
|
|
await this.setStateAsync({
|
|
paymentName: paymentName,
|
|
});
|
|
this.onUpdatePricing(this.state.pricing);
|
|
}
|
|
const res = await PaymentBackend.getPayment(this.state.owner, this.state.paymentName);
|
|
if (res.status !== "ok") {
|
|
throw new Error(res.msg);
|
|
}
|
|
const payment = res.data;
|
|
await this.setStateAsync({
|
|
payment: payment,
|
|
});
|
|
if (payment.state === "Created") {
|
|
if (["PayPal", "Stripe", "Alipay"].includes(payment.type)) {
|
|
this.setState({
|
|
timeout: setTimeout(async() => {
|
|
await PaymentBackend.notifyPayment(this.state.owner, this.state.paymentName);
|
|
this.getPayment();
|
|
}, 1000),
|
|
});
|
|
} else {
|
|
this.setState({
|
|
timeout: setTimeout(() => this.getPayment(), 1000),
|
|
});
|
|
}
|
|
}
|
|
} catch (err) {
|
|
Setting.showMessage("error", err.message);
|
|
return;
|
|
}
|
|
}
|
|
|
|
goToPaymentUrl(payment) {
|
|
if (payment.returnUrl === undefined || payment.returnUrl === null || payment.returnUrl === "") {
|
|
Setting.goToLink(`${window.location.origin}/products/${payment.owner}/${payment.productName}/buy`);
|
|
} else {
|
|
Setting.goToLink(payment.returnUrl);
|
|
}
|
|
}
|
|
|
|
render() {
|
|
const payment = this.state.payment;
|
|
|
|
if (payment === null) {
|
|
return null;
|
|
}
|
|
|
|
if (payment.state === "Paid") {
|
|
return (
|
|
<div className="login-content">
|
|
{
|
|
Setting.renderHelmet(payment)
|
|
}
|
|
<Result
|
|
status="success"
|
|
title={`${i18next.t("payment:You have successfully completed the payment")}: ${payment.productDisplayName}`}
|
|
subTitle={i18next.t("payment:Please click the below button to return to the original website")}
|
|
extra={[
|
|
<Button type="primary" key="returnUrl" onClick={() => {
|
|
this.goToPaymentUrl(payment);
|
|
}}>
|
|
{i18next.t("payment:Return to Website")}
|
|
</Button>,
|
|
]}
|
|
/>
|
|
</div>
|
|
);
|
|
} else if (payment.state === "Created") {
|
|
return (
|
|
<div className="login-content">
|
|
{
|
|
Setting.renderHelmet(payment)
|
|
}
|
|
<Result
|
|
status="info"
|
|
title={`${i18next.t("payment:The payment is still under processing")}: ${payment.productDisplayName}, ${i18next.t("payment:the current state is")}: ${payment.state}, ${i18next.t("payment:please wait for a few seconds...")}`}
|
|
subTitle={i18next.t("payment:Please click the below button to return to the original website")}
|
|
extra={[
|
|
<Spin key="returnUrl" size="large" tip={i18next.t("payment:Processing...")} />,
|
|
]}
|
|
/>
|
|
</div>
|
|
);
|
|
} else if (payment.state === "Canceled") {
|
|
return (
|
|
<div className="login-content">
|
|
{
|
|
Setting.renderHelmet(payment)
|
|
}
|
|
<Result
|
|
status="warning"
|
|
title={`${i18next.t("payment:The payment has been canceled")}: ${payment.productDisplayName}, ${i18next.t("payment:the current state is")}: ${payment.state}`}
|
|
subTitle={i18next.t("payment:Please click the below button to return to the original website")}
|
|
extra={[
|
|
<Button type="primary" key="returnUrl" onClick={() => {
|
|
this.goToPaymentUrl(payment);
|
|
}}>
|
|
{i18next.t("payment:Return to Website")}
|
|
</Button>,
|
|
]}
|
|
/>
|
|
</div>
|
|
);
|
|
} else if (payment.state === "Timeout") {
|
|
return (
|
|
<div className="login-content">
|
|
{
|
|
Setting.renderHelmet(payment)
|
|
}
|
|
<Result
|
|
status="warning"
|
|
title={`${i18next.t("payment:The payment has time out")}: ${payment.productDisplayName}, ${i18next.t("payment:the current state is")}: ${payment.state}`}
|
|
subTitle={i18next.t("payment:Please click the below button to return to the original website")}
|
|
extra={[
|
|
<Button type="primary" key="returnUrl" onClick={() => {
|
|
this.goToPaymentUrl(payment);
|
|
}}>
|
|
{i18next.t("payment:Return to Website")}
|
|
</Button>,
|
|
]}
|
|
/>
|
|
</div>
|
|
);
|
|
} else {
|
|
return (
|
|
<div className="login-content">
|
|
{
|
|
Setting.renderHelmet(payment)
|
|
}
|
|
<Result
|
|
status="error"
|
|
title={`${i18next.t("payment:The payment has failed")}: ${payment.productDisplayName}, ${i18next.t("payment:the current state is")}: ${payment.state}`}
|
|
subTitle={`${i18next.t("payment:Failed reason")}: ${payment.message}`}
|
|
extra={[
|
|
<Button type="primary" key="returnUrl" onClick={() => {
|
|
this.goToPaymentUrl(payment);
|
|
}}>
|
|
{i18next.t("payment:Return to Website")}
|
|
</Button>,
|
|
]}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
export default PaymentResultPage;
|