feat: added user-history resolvers, service files and module

This commit is contained in:
ankitsridhar16
2022-12-20 14:35:25 +05:30
parent b9ade5d2a3
commit a28774c2c4
3 changed files with 288 additions and 0 deletions

View File

@@ -0,0 +1,14 @@
import { Module } from '@nestjs/common';
import { PrismaModule } from '../prisma/prisma.module';
import { PubSubModule } from '../pubsub/pubsub.module';
import { UserModule } from '../user/user.module';
import { UserHistoryUserResolver } from './user.resolver';
import { UserHistoryResolver } from './user-history.resolver';
import { UserHistoryService } from './user-history.service';
@Module({
imports: [PrismaModule, PubSubModule, UserModule],
providers: [UserHistoryResolver, UserHistoryService, UserHistoryUserResolver],
exports: [UserHistoryService],
})
export class UserHistoryModule {}

View File

@@ -0,0 +1,48 @@
import { Args, Mutation, Resolver } from '@nestjs/graphql';
import { UserHistoryService } from './user-history.service';
import { PubSubService } from '../pubsub/pubsub.service';
import { UserHistory } from './user-history.model';
import { UseGuards } from '@nestjs/common';
import { GqlAuthGuard } from '../guards/gql-auth.guard';
import { GqlUser } from '../decorators/gql-user.decorator';
import { User } from '../user/user.model';
@Resolver()
export class UserHistoryResolver {
constructor(
private readonly userHistoryService: UserHistoryService,
private readonly pubsub: PubSubService,
) {}
/* Mutations */
@Mutation(() => UserHistory, {
description: 'Adds a new REST/GQL request to user history',
})
@UseGuards(GqlAuthGuard)
async addRequestToHistory(
@GqlUser() user: User,
@Args({
name: 'reqData',
description: 'JSON string of the request data',
})
reqData: string,
@Args({
name: 'resMetadata',
description: 'JSON string of the response metadata',
})
resMetadata: string,
@Args({
name: 'reqType',
description: 'string that denotes type of request REST or GQL',
})
reqType: string,
): Promise<UserHistory> {
return await this.userHistoryService.addRequestToHistory(
user.uid,
reqData,
resMetadata,
reqType,
);
}
}

View File

@@ -0,0 +1,226 @@
import { Injectable } from '@nestjs/common';
import { PrismaService } from '../prisma/prisma.service';
import { PubSubService } from '../pubsub/pubsub.service';
import { ReqType, UserHistory } from './user-history.model';
import * as E from 'fp-ts/Either';
// Contains constants for the subscription types we send to pubsub service
enum SubscriptionType {
Created = 'created',
Updated = 'updated',
Deleted = 'deleted',
}
@Injectable()
export class UserHistoryService {
constructor(
private readonly prisma: PrismaService,
private readonly pubsub: PubSubService,
) {}
/**
* Fetch users REST or GraphQL history based on ReqType argument.
* @param uid Users uid
* @param reqType request Type to fetch i.e. GraphQL or REST
* @returns an array of user history
*/
async fetchUserHistory(uid: string, reqType: ReqType) {
const userHistory = await this.prisma.userHistory.findMany({
where: {
userUid: uid,
type: reqType,
},
});
const userHistoryColl: UserHistory[] = [];
userHistory.forEach((history) => {
userHistoryColl.push(<UserHistory>{
id: history.id,
userUid: history.userUid,
reqType: history.type,
request: JSON.stringify(history.request),
responseMetadata: JSON.stringify(history.responseMetadata),
isStarred: history.isStarred,
});
});
return userHistoryColl;
}
/**
* Adds a request to users history.
* @param uid Users uid
* @param reqData the request data
* @param resMetadata the response metadata
* @param reqType request Type to fetch i.e. GraphQL or REST
* @returns an array of user history
*/
async addRequestToHistory(
uid: string,
reqData: string,
resMetadata: string,
reqType: string,
) {
const requestType = this.validateReqType(reqType);
const history = await this.prisma.userHistory.create({
data: {
userUid: uid,
request: JSON.parse(reqData),
responseMetadata: JSON.parse(resMetadata),
type: requestType,
isStarred: false,
},
});
const userHistory = <UserHistory>{
id: history.id,
userUid: history.userUid,
request: JSON.stringify(history.request),
responseMetadata: JSON.stringify(history.responseMetadata),
executedOn: history.executedOn,
isStarred: history.isStarred,
reqType: history.type,
};
await this.publishUserHistorySubscription(
userHistory,
SubscriptionType.Created,
);
return userHistory;
}
/**
* Stars or unstars a request in the history
* @param uid Users uid
* @param id id of the request in the history
* @returns an Either of updated `UserHistory` or Error
*/
async starUnstarRequestInHistory(uid: string, id: string) {
const userHistory = await this.prisma.userHistory.findFirst({
where: {
id: id,
},
});
if (userHistory == null) {
return E.left('history doesnt exist');
}
try {
const updatedHistory = await this.prisma.userHistory.update({
where: {
id: id,
},
data: {
isStarred: !userHistory.isStarred,
},
});
const updatedUserHistory = <UserHistory>{
id: updatedHistory.id,
userUid: updatedHistory.userUid,
request: JSON.stringify(updatedHistory.request),
responseMetadata: JSON.stringify(updatedHistory.responseMetadata),
executedOn: updatedHistory.executedOn,
isStarred: updatedHistory.isStarred,
reqType: updatedHistory.type,
};
await this.publishUserHistorySubscription(
updatedUserHistory,
SubscriptionType.Updated,
);
return E.right(updatedUserHistory);
} catch (e) {
E.left('error updating');
}
}
/**
* Removes a REST/GraphQL request from the history
* @param uid Users uid
* @param id id of the request
* @returns an Either of deleted `UserHistory` or Error
*/
async removeRequestFromHistory(uid: string, id: string) {
try {
const delUserHistory = await this.prisma.userHistory.delete({
where: {
id: id,
},
});
const deletedUserHistory = <UserHistory>{
id: delUserHistory.id,
userUid: delUserHistory.userUid,
request: JSON.stringify(delUserHistory.request),
responseMetadata: JSON.stringify(delUserHistory.responseMetadata),
executedOn: delUserHistory.executedOn,
isStarred: delUserHistory.isStarred,
reqType: delUserHistory.type,
};
await this.publishUserHistorySubscription(
deletedUserHistory,
SubscriptionType.Deleted,
);
return E.right(deletedUserHistory);
} catch (e) {
return E.left('error deleting history not found');
}
}
/**
* Delete all REST/GraphQl user history based on ReqType
* @param uid Users uid
* @param reqType request type to be deleted i.e. REST or GraphQL
* @returns a count of deleted history
*/
async deleteAllUserHistory(uid: string, reqType: string) {
const requestType = this.validateReqType(reqType);
return await this.prisma.userHistory.deleteMany({
where: {
userUid: uid,
type: requestType,
},
});
}
// Method that takes a request type argument as string and validates against `ReqType`
validateReqType(reqType: string) {
let requestType: ReqType;
return reqType == ReqType.REST
? (requestType = ReqType.REST)
: (requestType = ReqType.GQL);
}
// Method to publish subscriptions based on the subscription type of the history
async publishUserHistorySubscription(
userHistory: UserHistory,
subscriptionType: SubscriptionType,
) {
switch (subscriptionType) {
case SubscriptionType.Created:
await this.pubsub.publish(
`user_history/${userHistory.id}/created`,
userHistory,
);
break;
case SubscriptionType.Updated:
await this.pubsub.publish(
`user_history/${userHistory.id}/updated`,
userHistory,
);
break;
case SubscriptionType.Deleted:
await this.pubsub.publish(
`user_history/${userHistory.id}/deleted`,
userHistory,
);
break;
default:
break;
}
}
}