Skip to content

Commit

Permalink
Merge branch 'master' into feature/email-ics-file-for-pickup-events
Browse files Browse the repository at this point in the history
  • Loading branch information
echang594 authored May 22, 2024
2 parents 1984dbb + 1ee1b5a commit f015b89
Show file tree
Hide file tree
Showing 4 changed files with 70 additions and 30 deletions.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@acmucsd/membership-portal",
"version": "3.6.1",
"version": "3.6.2",
"description": "REST API for ACM UCSD's membership portal.",
"main": "index.d.ts",
"files": [
Expand Down
12 changes: 10 additions & 2 deletions services/MerchStoreService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -536,10 +536,10 @@ export default class MerchStoreService {

// Verify that this order would not set the pickup event's order count
// over the order limit
const currentOrderCount = pickupEvent.orders.filter((o) => o.status !== OrderStatus.CANCELLED).length;
if (currentOrderCount >= pickupEvent.orderLimit) {
if (MerchStoreService.isPickupEventOrderLimitFull(pickupEvent)) {
throw new UserError('This merch pickup event is full! Please choose a different pickup event');
}

const totalCost = MerchStoreService.totalCost(originalOrder, itemOptions);
const merchOrderRepository = Repositories.merchOrder(txn);

Expand Down Expand Up @@ -683,6 +683,11 @@ export default class MerchStoreService {
return new Date() > moment(pickupEvent.start).subtract(2, 'days').toDate();
}

private static isPickupEventOrderLimitFull(pickupEvent: OrderPickupEventModel): boolean {
const currentOrderCount = pickupEvent.orders.filter((o) => o.status !== OrderStatus.CANCELLED).length;
return currentOrderCount >= pickupEvent.orderLimit;
}

/**
* Changes the pickup event of an order to a new one. The new pickup event must start more than 2 calendar
* days after the current time.
Expand All @@ -709,6 +714,9 @@ export default class MerchStoreService {
if (MerchStoreService.isLessThanTwoDaysBeforePickupEvent(newPickupEventForOrder)) {
throw new UserError('Cannot change order pickup to an event that starts in less than 2 days');
}
if (MerchStoreService.isPickupEventOrderLimitFull(newPickupEventForOrder)) {
throw new UserError('This merch pickup event is full! Please choose a different pickup event');
}
const orderInfo = await MerchStoreService.buildOrderUpdateInfo(order, newPickupEventForOrder, txn);
const calendarInfo = MerchStoreService.toEventCalendarInfo(newPickupEventForOrder);
await this.emailService.sendOrderPickupUpdated(user.email, user.firstName, orderInfo, calendarInfo);
Expand Down
30 changes: 3 additions & 27 deletions services/UserAuthService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,15 +6,15 @@ import { InjectManager } from 'typeorm-typedi-extensions';
import { EntityManager } from 'typeorm';
import { ExpressCheckinModel } from 'models/ExpressCheckinModel';
import { UserRepository } from '../repositories/UserRepository';
import { Uuid, ActivityType, UserState, UserRegistration } from '../types';
import { Uuid, ActivityType, UserState, UserRegistration, UserAccessType } from '../types';
import { Config } from '../config';
import { UserModel } from '../models/UserModel';
import Repositories, { TransactionsManager } from '../repositories';
import UserAccountService from './UserAccountService';

interface AuthToken {
uuid: Uuid;
admin: boolean;
accessType: UserAccessType;
}

@Service()
Expand Down Expand Up @@ -111,30 +111,6 @@ export default class UserAuthService {
return user;
}

public async login(email: string, pass: string): Promise<string> {
const authenticatedUser = await this.transactions.readWrite(async (txn) => {
let user = await Repositories
.user(txn)
.findByEmail(email.toLowerCase());
if (!user) throw new NotFoundError('There is no account associated with that email');
if (user.isBlocked()) throw new ForbiddenError('Your account has been blocked');
if (!(await user.verifyPass(pass))) throw new ForbiddenError('Incorrect password');
await Repositories.activity(txn).logActivity({
user,
type: ActivityType.ACCOUNT_LOGIN,
});
if (user.state === UserState.PASSWORD_RESET) {
user = await Repositories.user(txn).upsertUser(user, { state: UserState.ACTIVE });
}
return user;
});
const token: AuthToken = {
uuid: authenticatedUser.uuid,
admin: authenticatedUser.isAdmin(),
};
return jwt.sign(token, Config.auth.secret, { expiresIn: Config.auth.tokenLifespan });
}

public async checkCredentials(email: string, pass: string): Promise<UserModel> {
const authenticatedUser = await this.transactions.readWrite(async (txn) => {
const user = await Repositories
Expand All @@ -156,7 +132,7 @@ export default class UserAuthService {
public static generateAuthToken(user: UserModel): string {
const token: AuthToken = {
uuid: user.uuid,
admin: user.isAdmin(),
accessType: user.accessType,
};
return jwt.sign(token, Config.auth.secret, { expiresIn: Config.auth.tokenLifespan });
}
Expand Down
56 changes: 56 additions & 0 deletions tests/merchOrder.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1714,6 +1714,62 @@ describe('merch order pickup events', () => {
expect(persistedPickupEvent.pickupEvent.orderLimit).toEqual(2);
});

test('members cannot update their orders\' pickup events if the new pickup event is full', async () => {
const conn = await DatabaseConnection.get();
const member = UserFactory.fake({ credits: 10000 });
const item = MerchFactory.fakeItem({
hidden: false,
monthlyLimit: 100,
});
const option = MerchFactory.fakeOption({
item,
quantity: 2,
price: 2000,
});
const firstPickupEvent = MerchFactory.fakeFutureOrderPickupEvent({
orderLimit: 1,
});

const secondPickupEvent = MerchFactory.fakeFutureOrderPickupEvent({
orderLimit: 1,
});

await new PortalState()
.createUsers(member)
.createMerchItem(item)
.createMerchItemOptions(option)
.createOrderPickupEvents(firstPickupEvent, secondPickupEvent)
.orderMerch(member, [{ option, quantity: 1 }], firstPickupEvent)
.write();

const emailService = mock(EmailService);
when(emailService.sendOrderConfirmation(member.email, member.firstName, anything()))
.thenResolve();

// place order to secondPickupEvent
const order = [
{
option: option.uuid,
quantity: 1,
},
];
const placeMerchOrderRequest = {
order,
pickupEvent: secondPickupEvent.uuid,
};

const merchController = ControllerFactory.merchStore(conn, instance(emailService));
const placedOrderResponse = await merchController.placeMerchOrder(placeMerchOrderRequest, member);
const placedOrder = placedOrderResponse.order;

// attempt to reschedule to firstPickupEvent
const orderParams = { uuid: placedOrder.uuid };
const newPickupEventParams = { pickupEvent: firstPickupEvent.uuid };
await expect(merchController.rescheduleOrderPickup(orderParams, newPickupEventParams, member))
.rejects
.toThrow('This merch pickup event is full! Please choose a different pickup event');
});

test('placing an order with a pickup event that has reached the order limit fails', async () => {
const conn = await DatabaseConnection.get();
const member = UserFactory.fake({ points: 100 });
Expand Down

0 comments on commit f015b89

Please sign in to comment.