mirror of
https://github.com/OneUptime/oneuptime
synced 2024-11-22 15:24:55 +00:00
420 lines
11 KiB
TypeScript
420 lines
11 KiB
TypeScript
import UserMiddleware from "../Middleware/UserAuthorization";
|
|
import DatabaseService from "../Services/DatabaseService";
|
|
import CreateBy from "../Types/Database/CreateBy";
|
|
import GroupBy from "../Types/Database/GroupBy";
|
|
import Query from "../Types/Database/Query";
|
|
import Select from "../Types/Database/Select";
|
|
import Sort from "../Types/Database/Sort";
|
|
import Express, {
|
|
ExpressRequest,
|
|
ExpressResponse,
|
|
ExpressRouter,
|
|
NextFunction,
|
|
OneUptimeRequest,
|
|
} from "../Utils/Express";
|
|
import Response from "../Utils/Response";
|
|
import CommonAPI from "./CommonAPI";
|
|
import BaseModel from "Common/Models/DatabaseModels/DatabaseBaseModel/DatabaseBaseModel";
|
|
import DatabaseCommonInteractionProps from "Common/Types/BaseDatabase/DatabaseCommonInteractionProps";
|
|
import {
|
|
DEFAULT_LIMIT,
|
|
LIMIT_PER_PROJECT,
|
|
} from "Common/Types/Database/LimitMax";
|
|
import PartialEntity from "Common/Types/Database/PartialEntity";
|
|
import BadRequestException from "Common/Types/Exception/BadRequestException";
|
|
import { JSONObject } from "Common/Types/JSON";
|
|
import JSONFunctions from "Common/Types/JSONFunctions";
|
|
import ObjectID from "Common/Types/ObjectID";
|
|
import { UserPermission } from "Common/Types/Permission";
|
|
import PositiveNumber from "Common/Types/PositiveNumber";
|
|
|
|
export default class BaseAPI<
|
|
TBaseModel extends BaseModel,
|
|
TBaseService extends DatabaseService<BaseModel>,
|
|
> {
|
|
public entityType: { new (): TBaseModel };
|
|
|
|
public router: ExpressRouter;
|
|
public service: TBaseService;
|
|
|
|
public constructor(type: { new (): TBaseModel }, service: TBaseService) {
|
|
this.entityType = type;
|
|
const router: ExpressRouter = Express.getRouter();
|
|
// Create
|
|
router.post(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.createItem(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
// List
|
|
router.post(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/get-list`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.getList(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
// List
|
|
router.get(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/get-list`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.getList(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
// count
|
|
router.post(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/count`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.count(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
// Get Item
|
|
router.post(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/:id/get-item`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.getItem(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
// Get Item
|
|
router.get(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/:id/get-item`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.getItem(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
// Update
|
|
router.put(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/:id`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.updateItem(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
// Delete
|
|
router.delete(
|
|
`${new this.entityType().getCrudApiPath()?.toString()}/:id`,
|
|
UserMiddleware.getUserMiddleware,
|
|
async (req: ExpressRequest, res: ExpressResponse, next: NextFunction) => {
|
|
try {
|
|
await this.deleteItem(req, res);
|
|
} catch (err) {
|
|
next(err);
|
|
}
|
|
},
|
|
);
|
|
|
|
this.router = router;
|
|
this.service = service;
|
|
}
|
|
|
|
public async getPermissionsForTenant(
|
|
req: ExpressRequest,
|
|
): Promise<Array<UserPermission>> {
|
|
const permissions: Array<UserPermission> = [];
|
|
|
|
const props: DatabaseCommonInteractionProps =
|
|
await CommonAPI.getDatabaseCommonInteractionProps(req);
|
|
|
|
if (
|
|
props &&
|
|
props.userTenantAccessPermission &&
|
|
props.userTenantAccessPermission[props.tenantId?.toString() || ""]
|
|
) {
|
|
return (
|
|
props.userTenantAccessPermission[props.tenantId?.toString() || ""]
|
|
?.permissions || []
|
|
);
|
|
}
|
|
|
|
return permissions;
|
|
}
|
|
|
|
public getTenantId(req: ExpressRequest): ObjectID | null {
|
|
if ((req as OneUptimeRequest).tenantId) {
|
|
return (req as OneUptimeRequest).tenantId as ObjectID;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public async getList(
|
|
req: ExpressRequest,
|
|
res: ExpressResponse,
|
|
): Promise<void> {
|
|
await this.onBeforeList(req, res);
|
|
|
|
const skip: PositiveNumber = req.query["skip"]
|
|
? new PositiveNumber(req.query["skip"] as string)
|
|
: new PositiveNumber(0);
|
|
|
|
const limit: PositiveNumber = req.query["limit"]
|
|
? new PositiveNumber(req.query["limit"] as string)
|
|
: new PositiveNumber(DEFAULT_LIMIT);
|
|
|
|
if (limit.toNumber() > LIMIT_PER_PROJECT) {
|
|
throw new BadRequestException(
|
|
"Limit should be less than " + LIMIT_PER_PROJECT,
|
|
);
|
|
}
|
|
|
|
if (skip.toNumber() < 0) {
|
|
throw new BadRequestException(
|
|
"Skip should be greater than or equal to 0",
|
|
);
|
|
}
|
|
|
|
if (limit.toNumber() <= 0) {
|
|
throw new BadRequestException("Limit should be greater than 0");
|
|
}
|
|
|
|
let query: Query<BaseModel> = {};
|
|
let select: Select<BaseModel> = {};
|
|
let sort: Sort<BaseModel> = {};
|
|
let groupBy: GroupBy<BaseModel> | undefined;
|
|
|
|
if (req.body) {
|
|
query = JSONFunctions.deserialize(req.body["query"]) as Query<BaseModel>;
|
|
|
|
select = JSONFunctions.deserialize(
|
|
req.body["select"],
|
|
) as Select<BaseModel>;
|
|
|
|
sort = JSONFunctions.deserialize(req.body["sort"]) as Sort<BaseModel>;
|
|
|
|
groupBy = JSONFunctions.deserialize(
|
|
req.body["groupBy"],
|
|
) as GroupBy<BaseModel>;
|
|
}
|
|
|
|
const databaseProps: DatabaseCommonInteractionProps =
|
|
await CommonAPI.getDatabaseCommonInteractionProps(req);
|
|
|
|
const list: Array<BaseModel> = await this.service.findBy({
|
|
query,
|
|
select,
|
|
skip: skip,
|
|
limit: limit,
|
|
groupBy: groupBy,
|
|
sort: sort,
|
|
props: databaseProps,
|
|
});
|
|
|
|
const count: PositiveNumber = await this.service.countBy({
|
|
query,
|
|
props: databaseProps,
|
|
});
|
|
|
|
return Response.sendEntityArrayResponse(
|
|
req,
|
|
res,
|
|
list,
|
|
count,
|
|
this.entityType,
|
|
);
|
|
}
|
|
|
|
public async count(req: ExpressRequest, res: ExpressResponse): Promise<void> {
|
|
let query: Query<BaseModel> = {};
|
|
|
|
await this.onBeforeCount(req, res);
|
|
|
|
if (req.body) {
|
|
query = JSONFunctions.deserialize(req.body["query"]) as Query<BaseModel>;
|
|
}
|
|
|
|
const databaseProps: DatabaseCommonInteractionProps =
|
|
await CommonAPI.getDatabaseCommonInteractionProps(req);
|
|
|
|
const count: PositiveNumber = await this.service.countBy({
|
|
query,
|
|
props: databaseProps,
|
|
});
|
|
|
|
return Response.sendJsonObjectResponse(req, res, {
|
|
count: count.toNumber(),
|
|
});
|
|
}
|
|
|
|
public async getItem(
|
|
req: ExpressRequest,
|
|
res: ExpressResponse,
|
|
): Promise<void> {
|
|
const objectId: ObjectID = new ObjectID(req.params["id"] as string);
|
|
await this.onBeforeGet(req, res);
|
|
let select: Select<BaseModel> = {};
|
|
|
|
if (req.body) {
|
|
select = JSONFunctions.deserialize(
|
|
req.body["select"],
|
|
) as Select<BaseModel>;
|
|
}
|
|
|
|
const item: BaseModel | null = await this.service.findOneById({
|
|
id: objectId,
|
|
select,
|
|
props: await CommonAPI.getDatabaseCommonInteractionProps(req),
|
|
});
|
|
|
|
return Response.sendEntityResponse(req, res, item, this.entityType);
|
|
}
|
|
|
|
public async deleteItem(
|
|
req: ExpressRequest,
|
|
res: ExpressResponse,
|
|
): Promise<void> {
|
|
await this.onBeforeDelete(req, res);
|
|
const objectId: ObjectID = new ObjectID(req.params["id"] as string);
|
|
|
|
await this.service.deleteOneById({
|
|
id: objectId,
|
|
props: await CommonAPI.getDatabaseCommonInteractionProps(req),
|
|
});
|
|
|
|
return Response.sendEmptySuccessResponse(req, res);
|
|
}
|
|
|
|
public async updateItem(
|
|
req: ExpressRequest,
|
|
res: ExpressResponse,
|
|
): Promise<void> {
|
|
await this.onBeforeUpdate(req, res);
|
|
const objectId: ObjectID = new ObjectID(req.params["id"] as string);
|
|
const objectIdString: string = objectId.toString();
|
|
const body: JSONObject = req.body;
|
|
|
|
const item: PartialEntity<TBaseModel> = JSONFunctions.deserialize(
|
|
body["data"] as JSONObject,
|
|
) as PartialEntity<TBaseModel>;
|
|
|
|
delete (item as any)["_id"];
|
|
delete (item as any)["createdAt"];
|
|
delete (item as any)["updatedAt"];
|
|
|
|
await this.service.updateOneById({
|
|
id: new ObjectID(objectIdString),
|
|
data: item,
|
|
props: await CommonAPI.getDatabaseCommonInteractionProps(req),
|
|
});
|
|
|
|
return Response.sendEmptySuccessResponse(req, res);
|
|
}
|
|
|
|
public async createItem(
|
|
req: ExpressRequest,
|
|
res: ExpressResponse,
|
|
): Promise<void> {
|
|
await this.onBeforeCreate(req, res);
|
|
const body: JSONObject = req.body;
|
|
|
|
const item: TBaseModel = BaseModel.fromJSON<TBaseModel>(
|
|
body["data"] as JSONObject,
|
|
this.entityType,
|
|
) as TBaseModel;
|
|
|
|
const miscDataProps: JSONObject = JSONFunctions.deserialize(
|
|
body["miscDataProps"] as JSONObject,
|
|
);
|
|
|
|
const createBy: CreateBy<TBaseModel> = {
|
|
data: item,
|
|
miscDataProps: miscDataProps,
|
|
props: await CommonAPI.getDatabaseCommonInteractionProps(req),
|
|
};
|
|
|
|
const savedItem: BaseModel = await this.service.create(createBy);
|
|
|
|
return Response.sendEntityResponse(req, res, savedItem, this.entityType);
|
|
}
|
|
|
|
public getRouter(): ExpressRouter {
|
|
return this.router;
|
|
}
|
|
|
|
public getEntityName(): string {
|
|
return this.entityType.name;
|
|
}
|
|
|
|
protected async onBeforeList(
|
|
_req: ExpressRequest,
|
|
_res: ExpressResponse,
|
|
): Promise<any> {
|
|
return Promise.resolve(true);
|
|
}
|
|
|
|
protected async onBeforeCreate(
|
|
_req: ExpressRequest,
|
|
_res: ExpressResponse,
|
|
): Promise<any> {
|
|
return Promise.resolve(true);
|
|
}
|
|
|
|
protected async onBeforeGet(
|
|
_req: ExpressRequest,
|
|
_res: ExpressResponse,
|
|
): Promise<any> {
|
|
return Promise.resolve(true);
|
|
}
|
|
|
|
protected async onBeforeUpdate(
|
|
_req: ExpressRequest,
|
|
_res: ExpressResponse,
|
|
): Promise<any> {
|
|
return Promise.resolve(true);
|
|
}
|
|
|
|
protected async onBeforeDelete(
|
|
_req: ExpressRequest,
|
|
_res: ExpressResponse,
|
|
): Promise<any> {
|
|
return Promise.resolve(true);
|
|
}
|
|
|
|
protected async onBeforeCount(
|
|
_req: ExpressRequest,
|
|
_res: ExpressResponse,
|
|
): Promise<any> {
|
|
return Promise.resolve(true);
|
|
}
|
|
}
|