mirror of
https://github.com/OneUptime/oneuptime.git
synced 2026-01-16 23:00:51 +00:00
167 lines
5.3 KiB
TypeScript
167 lines
5.3 KiB
TypeScript
import { IsBillingEnabled } from "../EnvironmentConfig";
|
|
import UserMiddleware from "../Middleware/UserAuthorization";
|
|
import BillingInvoiceService, {
|
|
Service as BillingInvoiceServiceType,
|
|
} from "../Services/BillingInvoiceService";
|
|
import BillingService, { Invoice } from "../Services/BillingService";
|
|
import ProjectService from "../Services/ProjectService";
|
|
import {
|
|
ExpressRequest,
|
|
ExpressResponse,
|
|
NextFunction,
|
|
OneUptimeRequest,
|
|
} from "../Utils/Express";
|
|
import Response from "../Utils/Response";
|
|
import BaseAPI from "./BaseAPI";
|
|
import BaseModel from "../../Models/DatabaseModels/DatabaseBaseModel/DatabaseBaseModel";
|
|
import BadDataException from "../../Types/Exception/BadDataException";
|
|
import { JSONObject } from "../../Types/JSON";
|
|
import Permission, { UserPermission } from "../../Types/Permission";
|
|
import BillingInvoice, {
|
|
InvoiceStatus,
|
|
} from "../../Models/DatabaseModels/BillingInvoice";
|
|
import Project from "../../Models/DatabaseModels/Project";
|
|
|
|
export default class UserAPI extends BaseAPI<
|
|
BillingInvoice,
|
|
BillingInvoiceServiceType
|
|
> {
|
|
public constructor() {
|
|
super(BillingInvoice, BillingInvoiceService);
|
|
|
|
this.router.post(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/pay`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
if (!IsBillingEnabled) {
|
|
throw new BadDataException(
|
|
"Billing is not enabled for this server",
|
|
);
|
|
}
|
|
|
|
if (req.body["projectId"]) {
|
|
throw new BadDataException("projectId is required in request body");
|
|
}
|
|
|
|
const userPermissions: Array<UserPermission> = (
|
|
await this.getPermissionsForTenant(req)
|
|
).filter((permission: UserPermission) => {
|
|
return (
|
|
permission.permission.toString() ===
|
|
Permission.ProjectOwner.toString() ||
|
|
permission.permission.toString() ===
|
|
Permission.EditInvoices.toString()
|
|
);
|
|
});
|
|
|
|
if (
|
|
userPermissions.length === 0 &&
|
|
!(req as OneUptimeRequest).userAuthorization?.isMasterAdmin
|
|
) {
|
|
throw new BadDataException(
|
|
`You need ${Permission.ProjectOwner} or ${Permission.EditInvoices} permission to pay invoices.`,
|
|
);
|
|
}
|
|
|
|
const project: Project | null = await ProjectService.findOneById({
|
|
id: this.getTenantId(req)!,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
select: {
|
|
_id: true,
|
|
paymentProviderCustomerId: true,
|
|
paymentProviderSubscriptionId: true,
|
|
paymentProviderMeteredSubscriptionId: true,
|
|
},
|
|
});
|
|
|
|
if (!project) {
|
|
throw new BadDataException("Project not found");
|
|
}
|
|
|
|
if (!project.paymentProviderCustomerId) {
|
|
throw new BadDataException("Payment Provider customer not found");
|
|
}
|
|
|
|
if (!project.paymentProviderSubscriptionId) {
|
|
throw new BadDataException(
|
|
"Payment Provider subscription not found",
|
|
);
|
|
}
|
|
|
|
const body: JSONObject = req.body;
|
|
|
|
const item: BillingInvoice = BaseModel.fromJSON<BillingInvoice>(
|
|
body["data"] as JSONObject,
|
|
this.entityType,
|
|
) as BillingInvoice;
|
|
|
|
if (!item.paymentProviderInvoiceId) {
|
|
throw new BadDataException("Invoice ID not found");
|
|
}
|
|
|
|
if (!item.paymentProviderCustomerId) {
|
|
throw new BadDataException("Customer ID not found");
|
|
}
|
|
let invoice: Invoice | null = null;
|
|
|
|
try {
|
|
invoice = await BillingService.payInvoice(
|
|
item.paymentProviderCustomerId!,
|
|
item.paymentProviderInvoiceId!,
|
|
);
|
|
} catch (err) {
|
|
invoice = await BillingService.getInvoice(
|
|
item.paymentProviderCustomerId!,
|
|
item.paymentProviderInvoiceId!,
|
|
);
|
|
|
|
// check if this invoice needs more authentication like 3ds secure.
|
|
if (
|
|
invoice.status === InvoiceStatus.Open &&
|
|
invoice.paymentIntentId
|
|
) {
|
|
const paymentIntentId: string = invoice.paymentIntentId;
|
|
|
|
const clientSecret: string =
|
|
await BillingService.getPaymentIntentClientSecret(
|
|
paymentIntentId,
|
|
);
|
|
|
|
return Response.sendJsonObjectResponse(req, res, {
|
|
clientSecret: clientSecret,
|
|
});
|
|
}
|
|
// otherwise we have no idea what went wrong and throw an error.
|
|
throw err;
|
|
}
|
|
|
|
// save updated status.
|
|
|
|
await this.service.updateOneBy({
|
|
query: {
|
|
paymentProviderInvoiceId: invoice.id!,
|
|
},
|
|
props: {
|
|
isRoot: true,
|
|
ignoreHooks: true,
|
|
},
|
|
data: {
|
|
status: invoice.status as InvoiceStatus,
|
|
},
|
|
});
|
|
|
|
await BillingInvoiceService.refreshSubscriptionStatus({
|
|
projectId: project.id!,
|
|
});
|
|
|
|
return Response.sendEmptySuccessResponse(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
}
|
|
}
|