Files
fluxer/fluxer_app/src/actions/GiftActionCreators.tsx
Hampus Kraft 2f557eda8c initial commit
2026-01-01 21:05:54 +00:00

221 lines
6.4 KiB
TypeScript

/*
* Copyright (C) 2026 Fluxer Contributors
*
* This file is part of Fluxer.
*
* Fluxer is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Fluxer is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Fluxer. If not, see <https://www.gnu.org/licenses/>.
*/
import type {I18n} from '@lingui/core';
import {msg} from '@lingui/core/macro';
import * as ModalActionCreators from '~/actions/ModalActionCreators';
import {modal} from '~/actions/ModalActionCreators';
import * as ToastActionCreators from '~/actions/ToastActionCreators';
import {APIErrorCodes} from '~/Constants';
import {GenericErrorModal} from '~/components/alerts/GenericErrorModal';
import {GiftAcceptModal} from '~/components/modals/GiftAcceptModal';
import {Endpoints} from '~/Endpoints';
import http, {HttpError} from '~/lib/HttpClient';
import {Logger} from '~/lib/Logger';
import type {UserPartial} from '~/records/UserRecord';
import DeveloperOptionsStore from '~/stores/DeveloperOptionsStore';
import GiftStore from '~/stores/GiftStore';
import UserStore from '~/stores/UserStore';
interface ApiErrorResponse {
code?: string;
message?: string;
errors?: Record<string, unknown>;
}
const logger = new Logger('Gifts');
export interface Gift {
code: string;
duration_months: number;
redeemed: boolean;
created_by?: UserPartial;
}
export interface GiftMetadata {
code: string;
duration_months: number;
created_at: string;
created_by: UserPartial;
redeemed_at: string | null;
redeemed_by: UserPartial | null;
}
export const fetch = async (code: string): Promise<Gift> => {
try {
const response = await http.get<Gift>({url: Endpoints.GIFT(code)});
const gift = response.body;
logger.debug('Gift fetched', {code});
return gift;
} catch (error) {
logger.error('Gift fetch failed', error);
if (error instanceof HttpError && error.status === 404) {
GiftStore.markAsInvalid(code);
}
throw error;
}
};
export const fetchWithCoalescing = async (code: string): Promise<Gift> => {
return GiftStore.fetchGift(code);
};
export const openAcceptModal = async (code: string): Promise<void> => {
void fetchWithCoalescing(code).catch(() => {});
ModalActionCreators.pushWithKey(
modal(() => <GiftAcceptModal code={code} />),
`gift-accept-${code}`,
);
};
export const redeem = async (i18n: I18n, code: string): Promise<void> => {
try {
await http.post({url: Endpoints.GIFT_REDEEM(code)});
logger.info('Gift redeemed', {code});
GiftStore.markAsRedeemed(code);
ToastActionCreators.success(i18n._(msg`Gift redeemed successfully!`));
} catch (error) {
logger.error('Gift redeem failed', error);
if (error instanceof HttpError) {
const errorResponse = error.body as ApiErrorResponse;
const errorCode = errorResponse?.code;
switch (errorCode) {
case APIErrorCodes.CANNOT_REDEEM_PLUTONIUM_WITH_VISIONARY:
ModalActionCreators.push(
modal(() => (
<GenericErrorModal
title={i18n._(msg`Cannot Redeem Gift`)}
message={i18n._(msg`You cannot redeem Plutonium gift codes while you have Visionary premium.`)}
/>
)),
);
break;
case APIErrorCodes.UNKNOWN_GIFT_CODE:
GiftStore.markAsInvalid(code);
ModalActionCreators.push(
modal(() => (
<GenericErrorModal
title={i18n._(msg`Invalid Gift Code`)}
message={i18n._(msg`This gift code is invalid or has already been redeemed.`)}
/>
)),
);
break;
case APIErrorCodes.GIFT_CODE_ALREADY_REDEEMED:
GiftStore.markAsRedeemed(code);
ModalActionCreators.push(
modal(() => (
<GenericErrorModal
title={i18n._(msg`Gift Already Redeemed`)}
message={i18n._(msg`This gift code has already been redeemed.`)}
/>
)),
);
break;
default:
if (error.status === 404) {
GiftStore.markAsInvalid(code);
ModalActionCreators.push(
modal(() => (
<GenericErrorModal
title={i18n._(msg`Gift Not Found`)}
message={i18n._(msg`This gift code could not be found.`)}
/>
)),
);
} else {
ModalActionCreators.push(
modal(() => (
<GenericErrorModal
title={i18n._(msg`Failed to Redeem Gift`)}
message={i18n._(msg`We couldn't redeem this gift code. Please try again.`)}
/>
)),
);
}
}
} else {
ModalActionCreators.push(
modal(() => (
<GenericErrorModal
title={i18n._(msg`Failed to Redeem Gift`)}
message={i18n._(msg`We couldn't redeem this gift code. Please try again.`)}
/>
)),
);
}
throw error;
}
};
export const fetchUserGifts = async (): Promise<Array<GiftMetadata>> => {
if (DeveloperOptionsStore.mockGiftInventory) {
const currentUser = UserStore.getCurrentUser();
const userPartial: UserPartial = currentUser
? {
id: currentUser.id,
username: currentUser.username,
discriminator: currentUser.discriminator,
avatar: currentUser.avatar,
flags: currentUser.flags,
}
: {
id: '000000000000000000',
username: 'MockUser',
discriminator: '0000',
avatar: null,
flags: 0,
};
const now = new Date();
const sevenDaysAgo = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000);
const twoDaysAgo = new Date(now.getTime() - 2 * 24 * 60 * 60 * 1000);
const durationMonths = DeveloperOptionsStore.mockGiftDurationMonths ?? 12;
const isRedeemed = DeveloperOptionsStore.mockGiftRedeemed ?? false;
const mockGift: GiftMetadata = {
code: 'MOCK-GIFT-TEST-1234',
duration_months: durationMonths,
created_at: sevenDaysAgo.toISOString(),
created_by: userPartial,
redeemed_at: isRedeemed ? twoDaysAgo.toISOString() : null,
redeemed_by: isRedeemed ? userPartial : null,
};
logger.debug('Returning mock user gifts', {count: 1});
return [mockGift];
}
try {
const response = await http.get<Array<GiftMetadata>>({url: Endpoints.USER_GIFTS});
const gifts = response.body;
logger.debug('User gifts fetched', {count: gifts.length});
return gifts;
} catch (error) {
logger.error('User gifts fetch failed', error);
throw error;
}
};