761 lines
35 KiB
TypeScript
761 lines
35 KiB
TypeScript
import axios, { type AxiosRequestConfig, type AxiosRequestHeaders, type AxiosResponse } from 'axios';
|
|
|
|
import type { ApiResponse } from '@/core/api.ts';
|
|
|
|
import type {
|
|
ApplicationCloudSetting
|
|
} from '@/core/setting.ts';
|
|
import type {
|
|
VersionInfo
|
|
} from '@/core/version.ts';
|
|
import {
|
|
TransactionType
|
|
} from '@/core/transaction.ts';
|
|
|
|
import {
|
|
BASE_API_URL_PATH,
|
|
BASE_QRCODE_PATH,
|
|
BASE_PROXY_URL_PATH,
|
|
BASE_AMAP_API_PROXY_URL_PATH,
|
|
DEFAULT_API_TIMEOUT,
|
|
DEFAULT_UPLOAD_API_TIMEOUT,
|
|
DEFAULT_EXPORT_API_TIMEOUT,
|
|
DEFAULT_IMPORT_API_TIMEOUT,
|
|
DEFAULT_CLEAR_ALL_TRANSACTIONS_API_TIMEOUT,
|
|
DEFAULT_LLM_API_TIMEOUT,
|
|
GOOGLE_MAP_JAVASCRIPT_URL,
|
|
BAIDU_MAP_JAVASCRIPT_URL,
|
|
AMAP_JAVASCRIPT_URL
|
|
} from '@/consts/api.ts';
|
|
|
|
import type {
|
|
AccountCreateRequest,
|
|
AccountModifyRequest,
|
|
AccountInfoResponse,
|
|
AccountHideRequest,
|
|
AccountMoveRequest,
|
|
AccountDeleteRequest
|
|
} from '@/models/account.ts';
|
|
import type {
|
|
AuthResponse,
|
|
RegisterResponse
|
|
} from '@/models/auth_response.ts';
|
|
import type {
|
|
ExportTransactionDataRequest,
|
|
ClearDataRequest,
|
|
ClearAccountTransactionsRequest,
|
|
DataStatisticsResponse
|
|
} from '@/models/data_management.ts';
|
|
import type {
|
|
UserCustomExchangeRateUpdateRequest,
|
|
UserCustomExchangeRateDeleteRequest,
|
|
UserCustomExchangeRateUpdateResponse,
|
|
LatestExchangeRateResponse
|
|
} from '@/models/exchange_rate.ts';
|
|
import type {
|
|
ForgetPasswordRequest
|
|
} from '@/models/forget_password.ts';
|
|
import type {
|
|
ImportTransactionResponsePageWrapper
|
|
} from '@/models/imported_transaction.ts';
|
|
import type {
|
|
TransactionCreateRequest,
|
|
TransactionModifyRequest,
|
|
TransactionDeleteRequest,
|
|
TransactionImportRequest,
|
|
TransactionListByMaxTimeRequest,
|
|
TransactionListInMonthByPageRequest,
|
|
TransactionInfoResponse,
|
|
TransactionInfoPageWrapperResponse,
|
|
TransactionInfoPageWrapperResponse2,
|
|
TransactionReconciliationStatementRequest,
|
|
TransactionReconciliationStatementResponse,
|
|
TransactionStatisticRequest,
|
|
TransactionStatisticResponse,
|
|
TransactionStatisticTrendsRequest,
|
|
TransactionStatisticTrendsResponseItem,
|
|
TransactionAmountsRequestParams,
|
|
TransactionAmountsResponse
|
|
} from '@/models/transaction.ts';
|
|
import {
|
|
TransactionAmountsRequest
|
|
} from '@/models/transaction.ts';
|
|
import type {
|
|
TransactionCategoryCreateRequest,
|
|
TransactionCategoryCreateBatchRequest,
|
|
TransactionCategoryModifyRequest,
|
|
TransactionCategoryHideRequest,
|
|
TransactionCategoryMoveRequest,
|
|
TransactionCategoryDeleteRequest,
|
|
TransactionCategoryInfoResponse
|
|
} from '@/models/transaction_category.ts';
|
|
import type {
|
|
TransactionPictureUnusedDeleteRequest,
|
|
TransactionPictureInfoBasicResponse
|
|
} from '@/models/transaction_picture_info.ts';
|
|
import type {
|
|
TransactionTagCreateRequest,
|
|
TransactionTagCreateBatchRequest,
|
|
TransactionTagModifyRequest,
|
|
TransactionTagHideRequest,
|
|
TransactionTagMoveRequest,
|
|
TransactionTagDeleteRequest,
|
|
TransactionTagInfoResponse
|
|
} from '@/models/transaction_tag.ts';
|
|
import type {
|
|
TransactionTemplateCreateRequest,
|
|
TransactionTemplateModifyRequest,
|
|
TransactionTemplateHideRequest,
|
|
TransactionTemplateMoveRequest,
|
|
TransactionTemplateDeleteRequest,
|
|
TransactionTemplateInfoResponse
|
|
} from '@/models/transaction_template.ts';
|
|
import type {
|
|
TokenGenerateMCPRequest,
|
|
TokenRevokeRequest,
|
|
TokenGenerateMCPResponse,
|
|
TokenRefreshResponse,
|
|
TokenInfoResponse
|
|
} from '@/models/token.ts';
|
|
import type {
|
|
TwoFactorEnableConfirmRequest,
|
|
TwoFactorEnableResponse,
|
|
TwoFactorEnableConfirmResponse,
|
|
TwoFactorDisableRequest,
|
|
TwoFactorRegenerateRecoveryCodeRequest,
|
|
TwoFactorStatusResponse
|
|
} from '@/models/two_factor.ts';
|
|
import type {
|
|
UserLoginRequest,
|
|
UserRegisterRequest,
|
|
UserVerifyEmailResponse,
|
|
UserResendVerifyEmailRequest,
|
|
UserProfileResponse,
|
|
UserProfileUpdateRequest,
|
|
UserProfileUpdateResponse
|
|
} from '@/models/user.ts';
|
|
import type {
|
|
UserApplicationCloudSettingsUpdateRequest
|
|
} from '@/models/user_app_cloud_setting.ts';
|
|
import type {
|
|
RecognizedReceiptImageResponse
|
|
} from '@/models/large_language_model.ts';
|
|
|
|
import {
|
|
getCurrentToken,
|
|
clearCurrentTokenAndUserInfo
|
|
} from './userstate.ts';
|
|
|
|
import {
|
|
isDefined,
|
|
isBoolean
|
|
} from './common.ts';
|
|
import {
|
|
getGoogleMapAPIKey,
|
|
getBaiduMapAK,
|
|
getAmapApplicationKey,
|
|
getExchangeRatesRequestTimeout
|
|
} from './server_settings.ts';
|
|
import { getTimezoneOffsetMinutes } from './datetime.ts';
|
|
import { generateRandomUUID } from './misc.ts';
|
|
import { getBasePath } from './web.ts';
|
|
|
|
interface ApiRequestConfig extends AxiosRequestConfig {
|
|
readonly headers: AxiosRequestHeaders;
|
|
readonly noAuth?: boolean;
|
|
readonly ignoreBlocked?: boolean;
|
|
readonly ignoreError?: boolean;
|
|
readonly timeout?: number;
|
|
}
|
|
|
|
export type ApiResponsePromise<T> = Promise<AxiosResponse<ApiResponse<T>>>;
|
|
|
|
let needBlockRequest = false;
|
|
const blockedRequests: ((token: string | undefined) => void)[] = [];
|
|
|
|
axios.defaults.baseURL = getBasePath() + BASE_API_URL_PATH;
|
|
axios.defaults.timeout = DEFAULT_API_TIMEOUT;
|
|
axios.interceptors.request.use((config: ApiRequestConfig) => {
|
|
const token = getCurrentToken();
|
|
|
|
if (token && !config.noAuth) {
|
|
config.headers.Authorization = `Bearer ${token}`;
|
|
}
|
|
|
|
config.headers['X-Timezone-Offset'] = getTimezoneOffsetMinutes();
|
|
|
|
if (needBlockRequest && !config.ignoreBlocked) {
|
|
return new Promise(resolve => {
|
|
blockedRequests.push(newToken => {
|
|
if (newToken) {
|
|
config.headers.Authorization = `Bearer ${newToken}`;
|
|
}
|
|
|
|
resolve(config);
|
|
});
|
|
});
|
|
}
|
|
|
|
return config;
|
|
}, error => {
|
|
return Promise.reject(error);
|
|
});
|
|
|
|
axios.interceptors.response.use(response => {
|
|
return response;
|
|
}, error => {
|
|
if (error.response && !error.response.config.ignoreError && error.response.data && error.response.data.errorCode) {
|
|
const errorCode = error.response.data.errorCode;
|
|
|
|
if (errorCode === 202001 // unauthorized access
|
|
|| errorCode === 202002 // current token is invalid
|
|
|| errorCode === 202003 // current token is expired
|
|
|| errorCode === 202004 // current token type is invalid
|
|
|| errorCode === 202005 // current token requires two-factor authorization
|
|
|| errorCode === 202006 // current token does not require two-factor authorization
|
|
|| errorCode === 202012 // token is empty
|
|
) {
|
|
clearCurrentTokenAndUserInfo(false);
|
|
location.reload();
|
|
return Promise.reject({ processed: true });
|
|
}
|
|
}
|
|
|
|
return Promise.reject(error);
|
|
});
|
|
|
|
export default {
|
|
setLocale: (locale: string) => {
|
|
axios.defaults.headers.common['Accept-Language'] = locale;
|
|
},
|
|
authorize: (data: UserLoginRequest): ApiResponsePromise<AuthResponse> => {
|
|
return axios.post<ApiResponse<AuthResponse>>('authorize.json', data);
|
|
},
|
|
authorize2FA: ({ passcode, token }: { passcode: string, token: string }): ApiResponsePromise<AuthResponse> => {
|
|
return axios.post<ApiResponse<AuthResponse>>('2fa/authorize.json', {
|
|
passcode: passcode
|
|
}, {
|
|
headers: {
|
|
Authorization: `Bearer ${token}`
|
|
}
|
|
});
|
|
},
|
|
authorize2FAByBackupCode: ({ recoveryCode, token }: { recoveryCode: string, token: string }): ApiResponsePromise<AuthResponse> => {
|
|
return axios.post<ApiResponse<AuthResponse>>('2fa/recovery.json', {
|
|
recoveryCode: recoveryCode
|
|
}, {
|
|
headers: {
|
|
Authorization: `Bearer ${token}`
|
|
}
|
|
});
|
|
},
|
|
register: (req: UserRegisterRequest): ApiResponsePromise<RegisterResponse> => {
|
|
return axios.post<ApiResponse<RegisterResponse>>('register.json', req);
|
|
},
|
|
verifyEmail: ({ token, requestNewToken }: { token: string, requestNewToken: boolean }): ApiResponsePromise<UserVerifyEmailResponse> => {
|
|
return axios.post<ApiResponse<UserVerifyEmailResponse>>('verify_email/by_token.json?token=' + token, {
|
|
requestNewToken: requestNewToken
|
|
}, {
|
|
noAuth: true,
|
|
ignoreError: true
|
|
} as ApiRequestConfig);
|
|
},
|
|
resendVerifyEmailByUnloginUser: (req: UserResendVerifyEmailRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('verify_email/resend.json', req);
|
|
},
|
|
requestResetPassword: (req: ForgetPasswordRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('forget_password/request.json', req);
|
|
},
|
|
resetPassword: ({ email, token, password }: { email: string, token: string, password: string }): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('forget_password/reset/by_token.json?token=' + token, {
|
|
email: email,
|
|
password: password
|
|
}, {
|
|
noAuth: true,
|
|
ignoreError: true
|
|
} as ApiRequestConfig);
|
|
},
|
|
logout: (): ApiResponsePromise<boolean> => {
|
|
return axios.get<ApiResponse<boolean>>('logout.json');
|
|
},
|
|
refreshToken: (): ApiResponsePromise<TokenRefreshResponse> => {
|
|
return new Promise((resolve) => {
|
|
needBlockRequest = true;
|
|
|
|
axios.post<ApiResponse<TokenRefreshResponse>>('v1/tokens/refresh.json', {}, {
|
|
ignoreBlocked: true
|
|
} as ApiRequestConfig).then(response => {
|
|
const data = response.data;
|
|
|
|
resolve(response);
|
|
needBlockRequest = false;
|
|
|
|
return data.result.newToken;
|
|
}).then(newToken => {
|
|
blockedRequests.forEach(func => func(newToken));
|
|
blockedRequests.length = 0;
|
|
});
|
|
});
|
|
},
|
|
getTokens: (): ApiResponsePromise<TokenInfoResponse[]> => {
|
|
return axios.get<ApiResponse<TokenInfoResponse[]>>('v1/tokens/list.json');
|
|
},
|
|
generateMCPToken: (req: TokenGenerateMCPRequest): ApiResponsePromise<TokenGenerateMCPResponse> => {
|
|
return axios.post<ApiResponse<TokenGenerateMCPResponse>>('v1/tokens/generate/mcp.json', req);
|
|
},
|
|
revokeToken: ({ tokenId, ignoreError }: { tokenId: string, ignoreError?: boolean }): ApiResponsePromise<boolean> => {
|
|
const req: TokenRevokeRequest = {
|
|
tokenId: tokenId
|
|
};
|
|
|
|
return axios.post<ApiResponse<boolean>>('v1/tokens/revoke.json', req, {
|
|
ignoreError: !!ignoreError
|
|
} as ApiRequestConfig);
|
|
},
|
|
revokeAllTokens: (): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/tokens/revoke_all.json');
|
|
},
|
|
getProfile: (): ApiResponsePromise<UserProfileResponse> => {
|
|
return axios.get<ApiResponse<UserProfileResponse>>('v1/users/profile/get.json');
|
|
},
|
|
updateProfile: (req: UserProfileUpdateRequest): ApiResponsePromise<UserProfileUpdateResponse> => {
|
|
return axios.post<ApiResponse<UserProfileUpdateResponse>>('v1/users/profile/update.json', req);
|
|
},
|
|
updateAvatar: ({ avatarFile }: { avatarFile: File }): ApiResponsePromise<UserProfileResponse> => {
|
|
return axios.postForm<ApiResponse<UserProfileResponse>>('v1/users/avatar/update.json', {
|
|
avatar: avatarFile
|
|
}, {
|
|
timeout: DEFAULT_UPLOAD_API_TIMEOUT
|
|
});
|
|
},
|
|
removeAvatar: (): ApiResponsePromise<UserProfileResponse> => {
|
|
return axios.post<ApiResponse<UserProfileResponse>>('v1/users/avatar/remove.json');
|
|
},
|
|
resendVerifyEmailByLoginedUser: (): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/users/verify_email/resend.json');
|
|
},
|
|
getUserApplicationCloudSettings: (): ApiResponsePromise<ApplicationCloudSetting[] | false> => {
|
|
return axios.get<ApiResponse<ApplicationCloudSetting[] | false>>('v1/users/settings/cloud/get.json');
|
|
},
|
|
updateUserApplicationCloudSettings: (req: UserApplicationCloudSettingsUpdateRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/users/settings/cloud/update.json', req);
|
|
},
|
|
disableUserApplicationCloudSettings: (): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/users/settings/cloud/disable.json');
|
|
},
|
|
get2FAStatus: (): ApiResponsePromise<TwoFactorStatusResponse> => {
|
|
return axios.get<ApiResponse<TwoFactorStatusResponse>>('v1/users/2fa/status.json');
|
|
},
|
|
enable2FA: (): ApiResponsePromise<TwoFactorEnableResponse> => {
|
|
return axios.post<ApiResponse<TwoFactorEnableResponse>>('v1/users/2fa/enable/request.json');
|
|
},
|
|
confirmEnable2FA: (req: TwoFactorEnableConfirmRequest): ApiResponsePromise<TwoFactorEnableConfirmResponse> => {
|
|
return axios.post<ApiResponse<TwoFactorEnableConfirmResponse>>('v1/users/2fa/enable/confirm.json', req);
|
|
},
|
|
disable2FA: (req: TwoFactorDisableRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/users/2fa/disable.json', req);
|
|
},
|
|
regenerate2FARecoveryCode: (req: TwoFactorRegenerateRecoveryCodeRequest): ApiResponsePromise<TwoFactorEnableConfirmResponse> => {
|
|
return axios.post<ApiResponse<TwoFactorEnableConfirmResponse>>('v1/users/2fa/recovery/regenerate.json', req);
|
|
},
|
|
getUserDataStatistics: (): ApiResponsePromise<DataStatisticsResponse> => {
|
|
return axios.get<ApiResponse<DataStatisticsResponse>>('v1/data/statistics.json');
|
|
},
|
|
getExportedUserData: (fileType: string, req?: ExportTransactionDataRequest): Promise<AxiosResponse<BlobPart>> => {
|
|
let params = '';
|
|
|
|
if (req) {
|
|
const amountFilter = encodeURIComponent(req.amountFilter);
|
|
const keyword = encodeURIComponent(req.keyword);
|
|
params = `max_time=${req.maxTime}&min_time=${req.minTime}&type=${req.type}&category_ids=${req.categoryIds}&account_ids=${req.accountIds}&tag_ids=${req.tagIds}&tag_filter_type=${req.tagFilterType}&amount_filter=${amountFilter}&keyword=${keyword}`;
|
|
} else {
|
|
params = 'max_time=0&min_time=0&type=0&category_ids=&account_ids=&tag_ids=&tag_filter_type=0&amount_filter=&keyword=';
|
|
}
|
|
|
|
if (fileType === 'csv') {
|
|
return axios.get<BlobPart>('v1/data/export.csv?' + params, {
|
|
timeout: DEFAULT_EXPORT_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
} else if (fileType === 'tsv') {
|
|
return axios.get<BlobPart>('v1/data/export.tsv?' + params, {
|
|
timeout: DEFAULT_EXPORT_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
} else {
|
|
return Promise.reject('Parameter Invalid');
|
|
}
|
|
},
|
|
clearAllData: (req: ClearDataRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/data/clear/all.json', req, {
|
|
timeout: DEFAULT_CLEAR_ALL_TRANSACTIONS_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
clearAllTransactions: (req: ClearDataRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/data/clear/transactions.json', req, {
|
|
timeout: DEFAULT_CLEAR_ALL_TRANSACTIONS_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
clearAllTransactionsOfAccount: (req: ClearAccountTransactionsRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/data/clear/transactions/by_account.json', req, {
|
|
timeout: DEFAULT_CLEAR_ALL_TRANSACTIONS_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
getAllAccounts: ({ visibleOnly }: { visibleOnly: boolean }): ApiResponsePromise<AccountInfoResponse[]> => {
|
|
return axios.get<ApiResponse<AccountInfoResponse[]>>('v1/accounts/list.json?visible_only=' + visibleOnly);
|
|
},
|
|
getAccount: ({ id }: { id: string }): ApiResponsePromise<AccountInfoResponse> => {
|
|
return axios.get<ApiResponse<AccountInfoResponse>>('v1/accounts/get.json?id=' + id);
|
|
},
|
|
addAccount: (req: AccountCreateRequest): ApiResponsePromise<AccountInfoResponse> => {
|
|
return axios.post<ApiResponse<AccountInfoResponse>>('v1/accounts/add.json', req);
|
|
},
|
|
modifyAccount: (req: AccountModifyRequest): ApiResponsePromise<AccountInfoResponse> => {
|
|
return axios.post<ApiResponse<AccountInfoResponse>>('v1/accounts/modify.json', req);
|
|
},
|
|
hideAccount: (req: AccountHideRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/accounts/hide.json', req);
|
|
},
|
|
moveAccount: (req: AccountMoveRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/accounts/move.json', req);
|
|
},
|
|
deleteAccount: (req: AccountDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/accounts/delete.json', req);
|
|
},
|
|
deleteSubAccount: (req: AccountDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/accounts/sub_account/delete.json', req);
|
|
},
|
|
getTransactions: (req: TransactionListByMaxTimeRequest): ApiResponsePromise<TransactionInfoPageWrapperResponse> => {
|
|
const amountFilter = encodeURIComponent(req.amountFilter);
|
|
const keyword = encodeURIComponent(req.keyword);
|
|
return axios.get<ApiResponse<TransactionInfoPageWrapperResponse>>(`v1/transactions/list.json?max_time=${req.maxTime}&min_time=${req.minTime}&type=${req.type}&category_ids=${req.categoryIds}&account_ids=${req.accountIds}&tag_ids=${req.tagIds}&tag_filter_type=${req.tagFilterType}&amount_filter=${amountFilter}&keyword=${keyword}&count=${req.count}&page=${req.page}&with_count=${req.withCount}&trim_account=true&trim_category=true&trim_tag=true`);
|
|
},
|
|
getAllTransactionsByMonth: (req: TransactionListInMonthByPageRequest): ApiResponsePromise<TransactionInfoPageWrapperResponse2> => {
|
|
const amountFilter = encodeURIComponent(req.amountFilter);
|
|
const keyword = encodeURIComponent(req.keyword);
|
|
return axios.get<ApiResponse<TransactionInfoPageWrapperResponse2>>(`v1/transactions/list/by_month.json?year=${req.year}&month=${req.month}&type=${req.type}&category_ids=${req.categoryIds}&account_ids=${req.accountIds}&tag_ids=${req.tagIds}&tag_filter_type=${req.tagFilterType}&amount_filter=${amountFilter}&keyword=${keyword}&trim_account=true&trim_category=true&trim_tag=true`);
|
|
},
|
|
getReconciliationStatements: (req: TransactionReconciliationStatementRequest): ApiResponsePromise<TransactionReconciliationStatementResponse> => {
|
|
return axios.get<ApiResponse<TransactionReconciliationStatementResponse>>(`v1/transactions/reconciliation_statements.json?account_id=${req.accountId}&start_time=${req.startTime}&end_time=${req.endTime}`);
|
|
},
|
|
getTransactionStatistics: (req: TransactionStatisticRequest): ApiResponsePromise<TransactionStatisticResponse> => {
|
|
const queryParams = [];
|
|
|
|
if (req.startTime) {
|
|
queryParams.push(`start_time=${req.startTime}`);
|
|
}
|
|
|
|
if (req.endTime) {
|
|
queryParams.push(`end_time=${req.endTime}`);
|
|
}
|
|
|
|
if (req.tagIds) {
|
|
queryParams.push(`tag_ids=${req.tagIds}`);
|
|
}
|
|
|
|
if (req.tagFilterType) {
|
|
queryParams.push(`tag_filter_type=${req.tagFilterType}`);
|
|
}
|
|
|
|
if (req.keyword) {
|
|
queryParams.push(`keyword=${encodeURIComponent(req.keyword)}`);
|
|
}
|
|
|
|
return axios.get<ApiResponse<TransactionStatisticResponse>>(`v1/transactions/statistics.json?use_transaction_timezone=${req.useTransactionTimezone}` + (queryParams.length ? '&' + queryParams.join('&') : ''));
|
|
},
|
|
getTransactionStatisticsTrends: (req: TransactionStatisticTrendsRequest): ApiResponsePromise<TransactionStatisticTrendsResponseItem[]> => {
|
|
const queryParams = [];
|
|
|
|
if (req.startYearMonth) {
|
|
queryParams.push(`start_year_month=${req.startYearMonth}`);
|
|
}
|
|
|
|
if (req.endYearMonth) {
|
|
queryParams.push(`end_year_month=${req.endYearMonth}`);
|
|
}
|
|
|
|
if (req.tagIds) {
|
|
queryParams.push(`tag_ids=${req.tagIds}`);
|
|
}
|
|
|
|
if (req.tagFilterType) {
|
|
queryParams.push(`tag_filter_type=${req.tagFilterType}`);
|
|
}
|
|
|
|
if (req.keyword) {
|
|
queryParams.push(`keyword=${encodeURIComponent(req.keyword)}`);
|
|
}
|
|
|
|
return axios.get<ApiResponse<TransactionStatisticTrendsResponseItem[]>>(`v1/transactions/statistics/trends.json?use_transaction_timezone=${req.useTransactionTimezone}` + (queryParams.length ? '&' + queryParams.join('&') : ''));
|
|
},
|
|
getTransactionAmounts: (params: TransactionAmountsRequestParams, excludeAccountIds: string[], excludeCategoryIds: string[]): ApiResponsePromise<TransactionAmountsResponse> => {
|
|
const req = TransactionAmountsRequest.of(params);
|
|
let queryParams = req.buildQuery();
|
|
|
|
if (excludeAccountIds && excludeAccountIds.length) {
|
|
queryParams = queryParams + `&exclude_account_ids=${excludeAccountIds.join(',')}`;
|
|
}
|
|
|
|
if (excludeCategoryIds && excludeCategoryIds.length) {
|
|
queryParams = queryParams + `&exclude_category_ids=${excludeCategoryIds.join(',')}`;
|
|
}
|
|
|
|
return axios.get<ApiResponse<TransactionAmountsResponse>>(`v1/transactions/amounts.json?${queryParams}`);
|
|
},
|
|
getTransaction: ({ id, withPictures }: { id: string, withPictures: boolean | undefined }): ApiResponsePromise<TransactionInfoResponse> => {
|
|
if (!isDefined(withPictures)) {
|
|
withPictures = true;
|
|
}
|
|
|
|
return axios.get<ApiResponse<TransactionInfoResponse>>(`v1/transactions/get.json?id=${id}&with_pictures=${withPictures}&trim_account=true&trim_category=true&trim_tag=true`);
|
|
},
|
|
addTransaction: (req: TransactionCreateRequest): ApiResponsePromise<TransactionInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionInfoResponse>>('v1/transactions/add.json', req);
|
|
},
|
|
modifyTransaction: (req: TransactionModifyRequest): ApiResponsePromise<TransactionInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionInfoResponse>>('v1/transactions/modify.json', req);
|
|
},
|
|
deleteTransaction: (req: TransactionDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transactions/delete.json', req);
|
|
},
|
|
parseImportDsvFile: ({ fileType, fileEncoding, importFile }: { fileType: string, fileEncoding?: string, importFile: File }): ApiResponsePromise<string[][]> => {
|
|
return axios.postForm<ApiResponse<string[][]>>('v1/transactions/parse_dsv_file.json', {
|
|
fileType: fileType,
|
|
fileEncoding: fileEncoding,
|
|
file: importFile
|
|
}, {
|
|
timeout: DEFAULT_UPLOAD_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
parseImportTransaction: ({ fileType, fileEncoding, importFile, columnMapping, transactionTypeMapping, hasHeaderLine, timeFormat, timezoneFormat, amountDecimalSeparator, amountDigitGroupingSymbol, geoSeparator, geoOrder, tagSeparator }: { fileType: string, fileEncoding?: string, importFile: File, columnMapping?: Record<number, number>, transactionTypeMapping?: Record<string, TransactionType>, hasHeaderLine?: boolean, timeFormat?: string, timezoneFormat?: string, amountDecimalSeparator?: string, amountDigitGroupingSymbol?: string, geoSeparator?: string, geoOrder?: string, tagSeparator?: string }): ApiResponsePromise<ImportTransactionResponsePageWrapper> => {
|
|
let textualColumnMapping: string | undefined = undefined;
|
|
let textualTransactionTypeMapping: string | undefined = undefined;
|
|
let textualHasHeaderLine: string | undefined = undefined;
|
|
|
|
if (columnMapping) {
|
|
textualColumnMapping = JSON.stringify(columnMapping);
|
|
}
|
|
|
|
if (transactionTypeMapping) {
|
|
textualTransactionTypeMapping = JSON.stringify(transactionTypeMapping);
|
|
}
|
|
|
|
if (hasHeaderLine) {
|
|
textualHasHeaderLine = 'true';
|
|
}
|
|
|
|
return axios.postForm<ApiResponse<ImportTransactionResponsePageWrapper>>('v1/transactions/parse_import.json', {
|
|
fileType: fileType,
|
|
fileEncoding: fileEncoding,
|
|
file: importFile,
|
|
columnMapping: textualColumnMapping,
|
|
transactionTypeMapping: textualTransactionTypeMapping,
|
|
hasHeaderLine: textualHasHeaderLine,
|
|
timeFormat: timeFormat,
|
|
timezoneFormat: timezoneFormat,
|
|
amountDecimalSeparator: amountDecimalSeparator,
|
|
amountDigitGroupingSymbol: amountDigitGroupingSymbol,
|
|
geoSeparator: geoSeparator,
|
|
geoOrder: geoOrder,
|
|
tagSeparator: tagSeparator
|
|
}, {
|
|
timeout: DEFAULT_UPLOAD_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
importTransactions: (req: TransactionImportRequest): ApiResponsePromise<number> => {
|
|
return axios.post<ApiResponse<number>>('v1/transactions/import.json', req, {
|
|
timeout: DEFAULT_IMPORT_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
getImportTransactionsProcess: (clientSessionId: string): ApiResponsePromise<number | null> => {
|
|
return axios.get<ApiResponse<number | null>>('v1/transactions/import/process.json?client_session_id=' + clientSessionId, {
|
|
ignoreError: true
|
|
} as ApiRequestConfig);
|
|
},
|
|
uploadTransactionPicture: ({ pictureFile, clientSessionId }: { pictureFile: File, clientSessionId?: string }): ApiResponsePromise<TransactionPictureInfoBasicResponse> => {
|
|
return axios.postForm<ApiResponse<TransactionPictureInfoBasicResponse>>('v1/transaction/pictures/upload.json', {
|
|
picture: pictureFile,
|
|
clientSessionId: clientSessionId
|
|
}, {
|
|
timeout: DEFAULT_UPLOAD_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
removeUnusedTransactionPicture: (req: TransactionPictureUnusedDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/pictures/remove_unused.json', req);
|
|
},
|
|
getAllTransactionCategories: (): ApiResponsePromise<Record<number, TransactionCategoryInfoResponse[]>> => {
|
|
return axios.get<ApiResponse<Record<number, TransactionCategoryInfoResponse[]>>>('v1/transaction/categories/list.json');
|
|
},
|
|
getTransactionCategory: ({ id }: { id: string }): ApiResponsePromise<TransactionCategoryInfoResponse> => {
|
|
return axios.get<ApiResponse<TransactionCategoryInfoResponse>>('v1/transaction/categories/get.json?id=' + id);
|
|
},
|
|
addTransactionCategory: (req: TransactionCategoryCreateRequest): ApiResponsePromise<TransactionCategoryInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionCategoryInfoResponse>>('v1/transaction/categories/add.json', req);
|
|
},
|
|
addTransactionCategoryBatch: (req: TransactionCategoryCreateBatchRequest): ApiResponsePromise<Record<number, TransactionCategoryInfoResponse[]>> => {
|
|
return axios.post<ApiResponse<Record<number, TransactionCategoryInfoResponse[]>>>('v1/transaction/categories/add_batch.json', req);
|
|
},
|
|
modifyTransactionCategory: (req: TransactionCategoryModifyRequest): ApiResponsePromise<TransactionCategoryInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionCategoryInfoResponse>>('v1/transaction/categories/modify.json', req);
|
|
},
|
|
hideTransactionCategory: (req: TransactionCategoryHideRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/categories/hide.json', req);
|
|
},
|
|
moveTransactionCategory: (req: TransactionCategoryMoveRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/categories/move.json', req);
|
|
},
|
|
deleteTransactionCategory: (req: TransactionCategoryDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/categories/delete.json', req);
|
|
},
|
|
getAllTransactionTags: (): ApiResponsePromise<TransactionTagInfoResponse[]> => {
|
|
return axios.get<ApiResponse<TransactionTagInfoResponse[]>>('v1/transaction/tags/list.json');
|
|
},
|
|
getTransactionTag: ({ id }: { id: string }): ApiResponsePromise<TransactionTagInfoResponse> => {
|
|
return axios.get<ApiResponse<TransactionTagInfoResponse>>('v1/transaction/tags/get.json?id=' + id);
|
|
},
|
|
addTransactionTag: (req: TransactionTagCreateRequest): ApiResponsePromise<TransactionTagInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionTagInfoResponse>>('v1/transaction/tags/add.json', req);
|
|
},
|
|
addTransactionTagBatch: (req: TransactionTagCreateBatchRequest): ApiResponsePromise<TransactionTagInfoResponse[]> => {
|
|
return axios.post<ApiResponse<TransactionTagInfoResponse[]>>('v1/transaction/tags/add_batch.json', req);
|
|
},
|
|
modifyTransactionTag: (req: TransactionTagModifyRequest): ApiResponsePromise<TransactionTagInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionTagInfoResponse>>('v1/transaction/tags/modify.json', req);
|
|
},
|
|
hideTransactionTag: (req: TransactionTagHideRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/tags/hide.json', req);
|
|
},
|
|
moveTransactionTag: (req: TransactionTagMoveRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/tags/move.json', req);
|
|
},
|
|
deleteTransactionTag: (req: TransactionTagDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/tags/delete.json', req);
|
|
},
|
|
getAllTransactionTemplates: ({ templateType }: { templateType: number }): ApiResponsePromise<TransactionTemplateInfoResponse[]> => {
|
|
return axios.get<ApiResponse<TransactionTemplateInfoResponse[]>>('v1/transaction/templates/list.json?templateType=' + templateType);
|
|
},
|
|
getTransactionTemplate: ({ id }: { id: string }): ApiResponsePromise<TransactionTemplateInfoResponse> => {
|
|
return axios.get<ApiResponse<TransactionTemplateInfoResponse>>('v1/transaction/templates/get.json?id=' + id);
|
|
},
|
|
addTransactionTemplate: (req: TransactionTemplateCreateRequest): ApiResponsePromise<TransactionTemplateInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionTemplateInfoResponse>>('v1/transaction/templates/add.json', req);
|
|
},
|
|
modifyTransactionTemplate: (req: TransactionTemplateModifyRequest): ApiResponsePromise<TransactionTemplateInfoResponse> => {
|
|
return axios.post<ApiResponse<TransactionTemplateInfoResponse>>('v1/transaction/templates/modify.json', req);
|
|
},
|
|
hideTransactionTemplate: (req: TransactionTemplateHideRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/templates/hide.json', req);
|
|
},
|
|
moveTransactionTemplate: (req: TransactionTemplateMoveRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/templates/move.json', req);
|
|
},
|
|
deleteTransactionTemplate: (req: TransactionTemplateDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/transaction/templates/delete.json', req);
|
|
},
|
|
recognizeReceiptImage: ({ imageFile }: { imageFile: File }): ApiResponsePromise<RecognizedReceiptImageResponse> => {
|
|
return axios.postForm<ApiResponse<RecognizedReceiptImageResponse>>('v1/llm/transactions/recognize_receipt_image.json', {
|
|
image: imageFile
|
|
}, {
|
|
timeout: DEFAULT_LLM_API_TIMEOUT
|
|
});
|
|
},
|
|
getLatestExchangeRates: (param: { ignoreError?: boolean }): ApiResponsePromise<LatestExchangeRateResponse> => {
|
|
return axios.get<ApiResponse<LatestExchangeRateResponse>>('v1/exchange_rates/latest.json', {
|
|
ignoreError: !!param.ignoreError,
|
|
timeout: getExchangeRatesRequestTimeout() || DEFAULT_API_TIMEOUT
|
|
} as ApiRequestConfig);
|
|
},
|
|
updateUserCustomExchangeRate: (req: UserCustomExchangeRateUpdateRequest): ApiResponsePromise<UserCustomExchangeRateUpdateResponse> => {
|
|
return axios.post<ApiResponse<UserCustomExchangeRateUpdateResponse>>('v1/exchange_rates/user_custom/update.json', req);
|
|
},
|
|
deleteUserCustomExchangeRate: (req: UserCustomExchangeRateDeleteRequest): ApiResponsePromise<boolean> => {
|
|
return axios.post<ApiResponse<boolean>>('v1/exchange_rates/user_custom/delete.json', req);
|
|
},
|
|
getServerVersion: (): ApiResponsePromise<VersionInfo> => {
|
|
return axios.get<ApiResponse<VersionInfo>>('v1/systems/version.json');
|
|
},
|
|
generateQrCodeUrl: (qrCodeName: string): string => {
|
|
return `${getBasePath()}${BASE_QRCODE_PATH}/${qrCodeName}.png`;
|
|
},
|
|
generateMapProxyTileImageUrl: (mapProvider: string, language: string): string => {
|
|
const token = getCurrentToken();
|
|
let url = `${getBasePath()}${BASE_PROXY_URL_PATH}/map/tile/{z}/{x}/{y}.png?provider=${mapProvider}&token=${token}`;
|
|
|
|
if (language) {
|
|
url = url + `&language=${language}`;
|
|
}
|
|
|
|
return url;
|
|
},
|
|
generateMapProxyAnnotationImageUrl: (mapProvider: string, language: string): string => {
|
|
const token = getCurrentToken();
|
|
let url = `${getBasePath()}${BASE_PROXY_URL_PATH}/map/annotation/{z}/{x}/{y}.png?provider=${mapProvider}&token=${token}`;
|
|
|
|
if (language) {
|
|
url = url + `&language=${language}`;
|
|
}
|
|
|
|
return url;
|
|
},
|
|
generateGoogleMapJavascriptUrl: (language: string | undefined, callbackFnName: string): string => {
|
|
let url = `${GOOGLE_MAP_JAVASCRIPT_URL}?key=${getGoogleMapAPIKey()}&libraries=core,marker&callback=${callbackFnName}`;
|
|
|
|
if (language) {
|
|
url = url + `&language=${language}`;
|
|
}
|
|
|
|
return url;
|
|
},
|
|
generateBaiduMapJavascriptUrl: (callbackFnName: string): string => {
|
|
return `${BAIDU_MAP_JAVASCRIPT_URL}&ak=${getBaiduMapAK()}&callback=${callbackFnName}`;
|
|
},
|
|
generateAmapJavascriptUrl: (callbackFnName: string): string => {
|
|
return `${AMAP_JAVASCRIPT_URL}&key=${getAmapApplicationKey()}&plugin=AMap.ToolBar&callback=${callbackFnName}`;
|
|
},
|
|
generateAmapApiInternalProxyUrl: (): string => {
|
|
return `${window.location.origin}${getBasePath()}${BASE_AMAP_API_PROXY_URL_PATH}`;
|
|
},
|
|
getInternalAvatarUrlWithToken(avatarUrl: string, disableBrowserCache?: boolean | string): string {
|
|
if (!avatarUrl) {
|
|
return avatarUrl;
|
|
}
|
|
|
|
const params = [];
|
|
params.push('token=' + getCurrentToken());
|
|
|
|
if (disableBrowserCache) {
|
|
if (isBoolean(disableBrowserCache)) {
|
|
params.push('_nocache=' + generateRandomUUID());
|
|
} else {
|
|
params.push('_nocache=' + disableBrowserCache);
|
|
}
|
|
}
|
|
|
|
if (avatarUrl.indexOf('?') >= 0) {
|
|
return avatarUrl + '&' + params.join('&');
|
|
} else {
|
|
return avatarUrl + '?' + params.join('&');
|
|
}
|
|
},
|
|
getTransactionPictureUrlWithToken(pictureUrl: string, disableBrowserCache?: boolean | string): string {
|
|
if (!pictureUrl) {
|
|
return pictureUrl;
|
|
}
|
|
|
|
const params = [];
|
|
params.push('token=' + getCurrentToken());
|
|
|
|
if (disableBrowserCache) {
|
|
if (isBoolean(disableBrowserCache)) {
|
|
params.push('_nocache=' + generateRandomUUID());
|
|
} else {
|
|
params.push('_nocache=' + disableBrowserCache);
|
|
}
|
|
}
|
|
|
|
if (pictureUrl.indexOf('?') >= 0) {
|
|
return pictureUrl + '&' + params.join('&');
|
|
} else {
|
|
return pictureUrl + '?' + params.join('&');
|
|
}
|
|
}
|
|
};
|