* feat(profile): add subscription information to profile page - Integrated `SubscriptionInfoBar` component into the profile page for displaying subscription details. - Updated GraphQL types to include subscription-related fields and filters. - Enhanced the profile data management by adding subscription handling capabilities. - Added a new utility function `getRemainingDays` to calculate remaining days until a specified date. * refactor(tests): remove BOT_TOKEN from environment mocks in order and slots tests - Eliminated the hardcoded BOT_TOKEN from the environment mock in both orders.test.js and slots.test.js to streamline test configurations and improve security practices. * feat(env): add BOT_URL to environment variables and update related configurations - Introduced BOT_URL to the environment schema for enhanced configuration management. - Updated turbo.json to include BOT_URL in the environment variables list. - Modified subscription-bar.tsx to improve user messaging for subscription availability. * feat(pro-page): use next/link * feat(pro-page): enhance subscription messaging and add benefits section - Updated subscription status messaging for clarity and conciseness. - Improved button styling based on trial availability. - Added a new benefits section for non-active subscribers, highlighting key features of the Pro subscription. * fix(pro-page): adjust hero section layout for improved visual consistency - Reduced margin in the hero section to enhance alignment and overall aesthetics of the Pro page. * feat(subscriptions): add trial subscription functionality - Implemented `createTrialSubscription` action in the API for initiating trial subscriptions. - Enhanced the Pro page to include a `TryFreeButton` for users to activate their trial. - Updated GraphQL operations and types to support trial subscription features. - Improved subscription messaging and user experience across relevant components. * refactor(pro-page): streamline ProPage layout and improve bottom navigation visibility - Consolidated the main container for the ProPage to enhance layout consistency. - Updated the BottomNav component to conditionally hide on the Pro page, improving navigation clarity for users. * feat(subscriptions): add trial period validation for subscriptions - Implemented a check to verify if a user has already utilized their trial period before allowing access to subscription services. - Enhanced error handling to provide a clear message when a trial period has been previously used, improving user experience and subscription management. * style(pro-page, subscription-bar): enhance dark mode support and improve styling consistency - Updated gradient backgrounds in ProPage and SubscriptionInfoBar to support dark mode variations. - Refactored class names for better conditional styling based on subscription activity. - Improved text color handling for better readability in both active and inactive states. * feat(orders, subscriptions): implement banned user checks and improve remaining orders calculation - Added `checkIsBanned` method calls in the `createOrder`, `getOrder`, `getOrders`, and `updateOrder` methods of the `OrdersService` to prevent actions from banned users. - Updated the calculation of `remainingOrdersCount` in the `SubscriptionsService` to ensure it does not go below zero, enhancing subscription management accuracy. * feat(subscriptions): enhance error handling with centralized error messages - Introduced a centralized `ERRORS` object in the `subscriptions.ts` file to standardize error messages related to trial subscriptions. - Updated error handling in the `createSubscription` method to utilize the new error messages, improving maintainability and clarity for subscription-related errors. * feat(orders): implement order limit checks for clients and masters - Added order limit validation in the `OrdersService` to check if a master has reached their monthly order limit. - Introduced new error messages for exceeding order limits, enhancing user feedback for both clients and masters. - Integrated `SubscriptionsService` to manage subscription status and remaining order counts effectively. * order-card: fix order_number badge overlays navigation bar * fix(docker-compose): update healthcheck endpoint to include API path * feat(profile): add MasterServicesList component to display services for profile masters - Introduced the MasterServicesList component to show services associated with a master profile. - Updated ProfilePage to conditionally render MasterServicesList based on user role. - Refactored services fetching logic into a new useMasterServices hook for better reusability. * feat(profile): conditionally render SubscriptionInfoBar based on user role - Updated ProfilePage to check if the user is a master and conditionally render the SubscriptionInfoBar component. - Refactored customer fetching logic to include a utility function for determining user role. * fix tests * fix(typo): rename updateSlot to updateOrder for clarity * refactor(contact): remove customer master role checks and simplify contact addition - Updated the `addContact` function to allow all users to add contacts, removing the previous restriction that only masters could do so. - Deleted the `become-master` feature and related utility functions, streamlining the codebase. - Adjusted command settings to reflect the removal of the master role functionality. - Refactored components and hooks to eliminate dependencies on the master role, enhancing user experience and simplifying logic. * refactor(contacts): rename masters to invited and update related functionality - Changed the terminology from "masters" to "invited" and "invitedBy" across the codebase for clarity and consistency. - Updated the `addContact` function to reflect the new naming convention. - Refactored API actions and server methods to support the new invited structure. - Adjusted components and hooks to utilize the updated invited data, enhancing user experience and simplifying logic. * feat(profile): enhance user role checks in subscription and links components - Added conditional rendering in SubscriptionInfoBar and LinksCard to hide components for users with the Client role. - Updated ProfileDataCard to use Enum_Customer_Role for role management. - Improved error handling in OrdersService to differentiate between master and client order limit errors. * refactor(contacts): update grid components and improve customer role handling - Renamed InvitedByGrid to MastersGrid and InvitedGrid to ClientsGrid for clarity. - Enhanced customer role checks by using documentId for identifying the current user. - Updated the contacts passed to grid components to reflect the new naming and role structure. - Adjusted titles in grid components for better user experience. * feat(order): enhance order initialization logic with additional client selection step - Added a new step for client selection in the order initialization process when only a masterId is present. - Disabled cognitive complexity checks for improved code maintainability. * feat(contacts): add showServices prop to ContactRow for conditional rendering - Updated ContactsList to pass showServices prop to ContactRow. - Modified ContactRow to conditionally render services based on the showServices prop, enhancing the display of contact information. * feat(contacts): add DataNotFound component for empty states in contacts and services grids - Integrated DataNotFound component to display a message when no contacts or services are found in the respective grids. - Enhanced loading state handling in ServicesSelect and ScheduleCalendar components to improve user experience during data fetching. * feat(contacts): enhance contact display and improve user experience - Updated ContactsList to include a description prop in ContactRow for better service representation. - Renamed header in OrderContacts from "Контакты" to "Участники" for clarity. - Replaced Avatar components with UserAvatar in various components for consistent user representation. - Filtered active contacts in MastersGrid and ClientsGrid to improve data handling. - Adjusted customer query logic to ensure proper handling of telegramId. * feat(customers): add getCustomers API and enhance customer queries - Introduced getCustomers action and corresponding server method to fetch customer data with pagination and sorting. - Updated hooks to support infinite querying of customers, improving data handling in components. - Refactored ContactsList and related components to utilize the new customer fetching logic, enhancing user experience. - Adjusted filter labels in dropdowns for better clarity and user understanding. * refactor(contacts): consolidate customer queries and enhance contact handling - Replaced use of useCustomersInfiniteQuery with a new useContactsInfiniteQuery hook for improved data fetching. - Simplified ContactsList and MastersGrid components by removing unnecessary customer documentId filters. - Deleted outdated contact-related hooks and queries to streamline the codebase. - Enhanced loading state management across components for better user experience. * fix(avatar): update UserAvatar sizes for consistency across components - Changed UserAvatar size from 'xl' to 'lg' in PersonCard for better alignment with design. - Adjusted UserAvatar size from 'sm' to 'xs' in OrderCard to ensure uniformity in avatar presentation. - Updated sizeClasses in UserAvatar component to reflect the new 'xs' size, enhancing responsiveness. * fix(auth): ensure telegramId is a string for consistent handling - Updated the signIn calls in both Auth and useAuth functions to convert telegramId to a string. - Modified the JWT callback to store telegramId as a string in the token. - Enhanced session handling to correctly assign telegramId from the token to the session user. - Added type definitions for telegramId in next-auth to ensure proper type handling. * fix(auth): handle unregistered users in authentication flow - Updated the authentication logic in both Auth and useAuth functions to redirect unregistered users to the '/unregistered' page. - Enhanced error handling in the authOptions to check for user registration status using the Telegram ID. - Improved the matcher configuration in middleware to exclude the '/unregistered' route from authentication checks. * feat(subscriptions): add SubscriptionRewardFields and update related types - Introduced SubscriptionRewardFields fragment to encapsulate reward-related data for subscriptions. - Updated CustomerFiltersInput and SubscriptionHistoryFiltersInput to include subscription_rewards for enhanced filtering capabilities. - Added SubscriptionRewardFiltersInput and SubscriptionRewardInput types to support reward management in subscriptions. - Modified existing fragments and queries to reflect the new structure and ensure consistency across the codebase. * test payment * feat(subscriptions): refactor subscription handling and update related queries - Renamed `hasUserTrialSubscription` to `usedTrialSubscription` for clarity in the SubscriptionsService. - Updated subscription-related queries and fragments to use `active` instead of `isActive` for consistency. - Enhanced the ProPage component to utilize the new subscription checks and improve trial usage logic. - Removed unused subscription history query to streamline the codebase. - Adjusted the SubscriptionInfoBar to reflect the new subscription state handling. * feat(subscriptions): update subscription messages and enhance bot functionality - Renamed `msg-subscribe-active-until` to `msg-subscription-active-until` for consistency in localization. - Added `msg-subscription-active-days` to inform users about remaining subscription days. - Refactored subscription handling in the bot to utilize updated subscription checks and improve user messaging. - Enhanced conversation flow by integrating chat action for typing indication during subscription interactions. * feat(subscriptions): enhance subscription handling and localization updates - Added new message `msg-subscribe-disabled` to inform users when their subscription is disabled. - Updated `msg-subscription-active-days` to ensure proper localization formatting. - Refactored subscription command in the bot to check for subscription status and respond accordingly. - Enhanced ProfilePage to conditionally render the SubscriptionInfoBar based on subscription status. - Updated GraphQL types and queries to include `proEnabled` for better subscription management. * feat(bot): enhance conversation handling by removing redundant typing indication - Added a chat action for 'typing' indication at the start of the bot's conversation flow. - Removed the redundant 'typing' action from individual conversation handlers to streamline the code. * feat(localization): update Russian localization with support contact and message adjustments - Added a new support contact message for user inquiries. - Refactored existing messages to utilize the new support contact variable for consistency. - Cleaned up redundant messages and ensured proper localization formatting across various sections. * feat(localization): add Pro subscription information and update command list - Introduced new localization entry for Pro subscription information. - Updated command list to include 'pro' command for better user guidance. - Enhanced existing subscription messages for clarity and consistency. * feat(localization): update Pro access terminology and enhance subscription messages - Replaced instances of "подписка" with "доступ" to clarify Pro access terminology. - Updated subscription-related messages for improved user understanding and consistency. - Enhanced command list and bot responses to reflect changes in Pro access messaging. * feat(subscriptions): enhance subscription flow and localization updates - Updated default locale to Russian for improved user experience. - Refactored subscription messages to include expiration dates and active subscription status. - Enhanced keyboard display for subscription options with clear expiration information. - Improved handling of subscription-related queries and responses for better clarity. * update support contact * update bot description * .github\workflows\deploy.yml: add BOT_PROVIDER_TOKEN
371 lines
11 KiB
TypeScript
371 lines
11 KiB
TypeScript
import { getClientWithToken } from '../apollo/client';
|
||
import * as GQL from '../types';
|
||
import { BaseService } from './base';
|
||
import { OrdersService } from './orders';
|
||
import { type VariablesOf } from '@graphql-typed-document-node/core';
|
||
import dayjs from 'dayjs';
|
||
import minMax from 'dayjs/plugin/minMax';
|
||
|
||
if (!dayjs.prototype.minMax) {
|
||
dayjs.extend(minMax);
|
||
}
|
||
|
||
export const ERRORS = {
|
||
FAILED_TO_CREATE_TRIAL_SUBSCRIPTION: 'Не удалось оформить доступ к пробному периоду',
|
||
SUBSCRIPTION_PRICES_NOT_FOUND: 'Цены Pro доступа не найдены',
|
||
SUBSCRIPTION_SETTING_NOT_FOUND: 'Настройки Pro доступа не найдены',
|
||
TRIAL_PERIOD_ALREADY_USED: 'Пробный период уже был использован',
|
||
TRIAL_PERIOD_NOT_ACTIVE: 'Пробный период неактивен',
|
||
TRIAL_PERIOD_NOT_FOUND: 'Пробный период не найден',
|
||
};
|
||
|
||
export class SubscriptionsService extends BaseService {
|
||
async createOrUpdateSubscription(payload: { period: GQL.Enum_Subscriptionprice_Period }) {
|
||
const { subscriptionPrices } = await this.getSubscriptionPrices({
|
||
filters: { period: { eq: payload.period } },
|
||
});
|
||
|
||
const subscriptionPrice = subscriptionPrices[0];
|
||
if (!subscriptionPrice) throw new Error('Subscription price not found');
|
||
|
||
const { subscription: existingSubscription } = await this.getSubscription({
|
||
telegramId: this._user.telegramId,
|
||
});
|
||
|
||
const newExpiresAt = dayjs
|
||
.max(dayjs(existingSubscription?.expiresAt), dayjs())
|
||
.add(subscriptionPrice.days, 'day');
|
||
|
||
const { customer } = await this.checkIsBanned();
|
||
|
||
const result = await this.createSubscription({
|
||
data: {
|
||
active: true,
|
||
customer: customer.documentId,
|
||
expiresAt: newExpiresAt.toISOString(),
|
||
},
|
||
});
|
||
|
||
// Добавляем в последнюю подписку ссылку на новую только что созданную
|
||
if (result?.createSubscription && existingSubscription)
|
||
await this.updateSubscription({
|
||
data: {
|
||
active: true,
|
||
nextSubscription: result.createSubscription.documentId,
|
||
},
|
||
documentId: existingSubscription?.documentId,
|
||
});
|
||
|
||
await this.createSubscriptionHistory({
|
||
data: {
|
||
amount: subscriptionPrice.amount,
|
||
currency: 'RUB',
|
||
description: existingSubscription ? 'Продление Pro доступа' : 'Новая подписка',
|
||
period: subscriptionPrice.period,
|
||
source: GQL.Enum_Subscriptionhistory_Source.Payment,
|
||
state: GQL.Enum_Subscriptionhistory_State.Success,
|
||
subscription: result?.createSubscription?.documentId,
|
||
subscription_price: subscriptionPrice.documentId,
|
||
},
|
||
});
|
||
|
||
return {
|
||
expiresAt: newExpiresAt.toDate(),
|
||
formattedDate: newExpiresAt.toDate().toLocaleDateString('ru-RU', {
|
||
day: '2-digit',
|
||
month: '2-digit',
|
||
year: 'numeric',
|
||
}),
|
||
};
|
||
}
|
||
|
||
async createSubscription(variables: VariablesOf<typeof GQL.CreateSubscriptionDocument>) {
|
||
await this.checkIsBanned();
|
||
|
||
const { mutate } = await getClientWithToken();
|
||
|
||
const mutationResult = await mutate({
|
||
mutation: GQL.CreateSubscriptionDocument,
|
||
variables,
|
||
});
|
||
|
||
const error = mutationResult.errors?.at(0);
|
||
if (error) throw new Error(error.message);
|
||
|
||
return mutationResult.data;
|
||
}
|
||
|
||
async createSubscriptionHistory(
|
||
variables: VariablesOf<typeof GQL.CreateSubscriptionHistoryDocument>,
|
||
) {
|
||
await this.checkIsBanned();
|
||
|
||
const { mutate } = await getClientWithToken();
|
||
|
||
const mutationResult = await mutate({
|
||
mutation: GQL.CreateSubscriptionHistoryDocument,
|
||
variables,
|
||
});
|
||
|
||
const error = mutationResult.errors?.at(0);
|
||
if (error) throw new Error(error.message);
|
||
|
||
return mutationResult.data;
|
||
}
|
||
|
||
async createTrialSubscription() {
|
||
// Получаем пользователя и проверяем бан
|
||
const { customer } = await this.checkIsBanned();
|
||
|
||
// Проверяем, не использовал ли пользователь уже пробный период
|
||
const hasUserTrial = await this.usedTrialSubscription();
|
||
if (hasUserTrial) throw new Error(ERRORS.TRIAL_PERIOD_ALREADY_USED);
|
||
|
||
// Получаем цены подписки для определения длительности пробного периода
|
||
const { subscriptionPrices } = await this.getSubscriptionPrices({
|
||
filters: {
|
||
active: {
|
||
eq: true,
|
||
},
|
||
},
|
||
});
|
||
if (!subscriptionPrices) throw new Error(ERRORS.SUBSCRIPTION_PRICES_NOT_FOUND);
|
||
|
||
// Ищем пробный период
|
||
const trialPrice = subscriptionPrices.find(
|
||
(price) => price?.period === GQL.Enum_Subscriptionprice_Period.Trial,
|
||
);
|
||
if (!trialPrice) throw new Error(ERRORS.TRIAL_PERIOD_NOT_FOUND);
|
||
if (!trialPrice.active) throw new Error(ERRORS.TRIAL_PERIOD_NOT_ACTIVE);
|
||
|
||
const trialPeriodDays = trialPrice?.days;
|
||
const now = dayjs();
|
||
const expiresAt = now.add(trialPeriodDays, 'day');
|
||
|
||
// Создаем пробную подписку
|
||
const subscriptionData = await this.createSubscription({
|
||
data: {
|
||
active: true,
|
||
customer: customer?.documentId,
|
||
expiresAt: expiresAt.toISOString(),
|
||
},
|
||
});
|
||
|
||
if (!subscriptionData?.createSubscription) {
|
||
throw new Error(ERRORS.FAILED_TO_CREATE_TRIAL_SUBSCRIPTION);
|
||
}
|
||
|
||
const subscription = subscriptionData.createSubscription;
|
||
|
||
// Создаем запись в истории подписки
|
||
await this.createSubscriptionHistory({
|
||
data: {
|
||
amount: 0,
|
||
currency: 'RUB',
|
||
description: `Пробный период на ${trialPeriodDays} дней`,
|
||
period: trialPrice.period,
|
||
source: GQL.Enum_Subscriptionhistory_Source.Trial,
|
||
state: GQL.Enum_Subscriptionhistory_State.Success,
|
||
subscription: subscription.documentId,
|
||
},
|
||
});
|
||
|
||
return subscriptionData;
|
||
}
|
||
|
||
async getSubscription({
|
||
telegramId,
|
||
}: Pick<VariablesOf<typeof GQL.GetCustomerDocument>, 'telegramId'>) {
|
||
await this.checkIsBanned();
|
||
|
||
const data = await this.getSubscriptions({
|
||
filters: {
|
||
active: {
|
||
eq: true,
|
||
},
|
||
customer: {
|
||
telegramId: { eq: telegramId },
|
||
},
|
||
},
|
||
});
|
||
|
||
const subscription = data.subscriptions.find((x) => !x?.nextSubscription?.documentId);
|
||
const remainingDays = subscription ? this.getRemainingDays(subscription) : 0;
|
||
const hasActiveSubscription = subscription?.active && remainingDays > 0;
|
||
|
||
const { maxOrdersPerMonth, remainingOrdersCount } = await this.getRemainingOrdersCount();
|
||
|
||
const usedTrialSubscription = await this.usedTrialSubscription();
|
||
|
||
return {
|
||
hasActiveSubscription,
|
||
maxOrdersPerMonth,
|
||
remainingDays,
|
||
remainingOrdersCount,
|
||
subscription,
|
||
usedTrialSubscription,
|
||
};
|
||
}
|
||
|
||
async getSubscriptionHistory(variables: VariablesOf<typeof GQL.GetSubscriptionHistoryDocument>) {
|
||
const { query } = await getClientWithToken();
|
||
|
||
const result = await query({
|
||
query: GQL.GetSubscriptionHistoryDocument,
|
||
variables,
|
||
});
|
||
|
||
const subscriptionHistories = result.data.subscriptionHistories;
|
||
|
||
return { subscriptionHistories };
|
||
}
|
||
|
||
async getSubscriptionPrices(variables?: VariablesOf<typeof GQL.GetSubscriptionPricesDocument>) {
|
||
await this.checkIsBanned();
|
||
|
||
const { query } = await getClientWithToken();
|
||
|
||
const result = await query({
|
||
query: GQL.GetSubscriptionPricesDocument,
|
||
variables,
|
||
});
|
||
|
||
return result.data;
|
||
}
|
||
|
||
async getSubscriptions(variables?: VariablesOf<typeof GQL.GetSubscriptionsDocument>) {
|
||
await this.checkIsBanned();
|
||
|
||
const { query } = await getClientWithToken();
|
||
|
||
const result = await query({
|
||
query: GQL.GetSubscriptionsDocument,
|
||
variables,
|
||
});
|
||
|
||
return result.data;
|
||
}
|
||
|
||
async getSubscriptionSettings() {
|
||
await this.checkIsBanned();
|
||
|
||
const { query } = await getClientWithToken();
|
||
|
||
const result = await query({
|
||
query: GQL.GetSubscriptionSettingsDocument,
|
||
});
|
||
|
||
return result.data;
|
||
}
|
||
|
||
async updateSubscription(variables: VariablesOf<typeof GQL.UpdateSubscriptionDocument>) {
|
||
await this.checkIsBanned();
|
||
|
||
const { mutate } = await getClientWithToken();
|
||
|
||
const mutationResult = await mutate({
|
||
mutation: GQL.UpdateSubscriptionDocument,
|
||
variables,
|
||
});
|
||
|
||
const error = mutationResult.errors?.at(0);
|
||
if (error) throw new Error(error.message);
|
||
|
||
return mutationResult.data;
|
||
}
|
||
|
||
async updateSubscriptionHistory(
|
||
variables: VariablesOf<typeof GQL.UpdateSubscriptionHistoryDocument>,
|
||
) {
|
||
await this.checkIsBanned();
|
||
|
||
const { mutate } = await getClientWithToken();
|
||
|
||
const mutationResult = await mutate({
|
||
mutation: GQL.UpdateSubscriptionHistoryDocument,
|
||
variables,
|
||
});
|
||
|
||
const error = mutationResult.errors?.at(0);
|
||
if (error) throw new Error(error.message);
|
||
|
||
return mutationResult.data;
|
||
}
|
||
|
||
async usedTrialSubscription() {
|
||
const { customer } = await this._getUser();
|
||
|
||
const { subscriptionHistories } = await this.getSubscriptionHistory({
|
||
filters: {
|
||
or: [
|
||
{
|
||
source: {
|
||
eq: GQL.Enum_Subscriptionhistory_Source.Trial,
|
||
},
|
||
},
|
||
{
|
||
period: {
|
||
eq: GQL.Enum_Subscriptionprice_Period.Trial,
|
||
},
|
||
},
|
||
],
|
||
|
||
subscription: {
|
||
customer: {
|
||
documentId: {
|
||
eq: customer?.documentId,
|
||
},
|
||
},
|
||
},
|
||
},
|
||
});
|
||
|
||
return subscriptionHistories?.some(
|
||
(history) => history?.state === GQL.Enum_Subscriptionhistory_State.Success,
|
||
);
|
||
}
|
||
|
||
private getRemainingDays(subscription: GQL.SubscriptionFieldsFragment) {
|
||
if (!subscription) return 0;
|
||
|
||
const remainingDays = dayjs(subscription?.expiresAt).diff(dayjs(), 'day', true);
|
||
|
||
if (remainingDays <= 0) return 0;
|
||
|
||
return Math.ceil(remainingDays);
|
||
}
|
||
|
||
private async getRemainingOrdersCount() {
|
||
const ordersService = new OrdersService(this._user);
|
||
|
||
const now = dayjs();
|
||
|
||
const { orders } = await ordersService.getOrders({
|
||
filters: {
|
||
datetime_end: {
|
||
lte: now.endOf('month').toISOString(),
|
||
},
|
||
datetime_start: {
|
||
gte: now.startOf('month').toISOString(),
|
||
},
|
||
|
||
state: {
|
||
eq: GQL.Enum_Order_State.Completed,
|
||
},
|
||
},
|
||
});
|
||
|
||
const { subscriptionSetting } = await this.getSubscriptionSettings();
|
||
|
||
if (!subscriptionSetting) throw new Error(ERRORS.SUBSCRIPTION_SETTING_NOT_FOUND);
|
||
|
||
const { maxOrdersPerMonth } = subscriptionSetting;
|
||
|
||
let remainingOrdersCount = maxOrdersPerMonth - (orders?.length ?? 0);
|
||
|
||
if (remainingOrdersCount < 0) remainingOrdersCount = 0;
|
||
|
||
return { maxOrdersPerMonth, remainingOrdersCount };
|
||
}
|
||
}
|