wallet-core/packages/merchant-backoffice-ui/tests/axiosMock.ts

446 lines
12 KiB
TypeScript

/*
This file is part of GNU Taler
(C) 2021 Taler Systems S.A.
GNU Taler is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
GNU Taler 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 General Public License for more details.
You should have received a copy of the GNU General Public License along with
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/
/**
*
* @author Sebastian Javier Marchano (sebasjm)
*/
import * as axios from 'axios';
import { MerchantBackend } from '../src/declaration';
import { mockAxiosOnce, setAxiosRequestAsTestingEnvironment } from '../src/utils/switchableAxios';
// import { mockAxiosOnce, setAxiosRequestAsTestingEnvironment } from "../src/hooks/backend";
export type Query<Req, Res> = (GetQuery | PostQuery | DeleteQuery | PatchQuery) & RequestResponse<Req, Res>
interface RequestResponse<Req, Res> {
code?: number,
}
interface GetQuery { get: string }
interface PostQuery { post: string }
interface DeleteQuery { delete: string }
interface PatchQuery { patch: string }
const JEST_DEBUG_LOG = process.env['JEST_DEBUG_LOG'] !== undefined
type ExpectationValues = { query: Query<any, any>; params?: { auth?: string, request?: any, qparam?: any, response?: any } }
type TestValues = [axios.AxiosRequestConfig | undefined, ExpectationValues | undefined]
const defaultCallback = (actualQuery?: axios.AxiosRequestConfig): axios.AxiosPromise<any> => {
if (JEST_DEBUG_LOG) {
console.log('UNEXPECTED QUERY', actualQuery)
}
throw Error('Default Axios mock callback is called, this mean that the test did a tried to use axios but there was no expectation in place, try using JEST_DEBUG_LOG env')
}
setAxiosRequestAsTestingEnvironment(
defaultCallback
);
export class AxiosMockEnvironment {
expectations: Array<{
query: Query<any, any>,
auth?: string,
params?: { request?: any, qparam?: any, response?: any },
result: { args: axios.AxiosRequestConfig | undefined }
} | undefined> = []
// axiosMock: jest.MockedFunction<axios.AxiosStatic>
addRequestExpectation<RequestType, ResponseType>(expectedQuery: Query<RequestType, ResponseType>, params: { auth?: string, request?: RequestType, qparam?: any, response?: ResponseType }): void {
const result = mockAxiosOnce(function (actualQuery?: axios.AxiosRequestConfig): axios.AxiosPromise {
if (JEST_DEBUG_LOG) {
console.log('query to the backend is made', actualQuery)
}
if (!expectedQuery) {
return Promise.reject("a query was made but it was not expected")
}
if (JEST_DEBUG_LOG) {
console.log('expected query:', params?.request)
console.log('expected qparams:', params?.qparam)
console.log('sending response:', params?.response)
}
const responseCode = expectedQuery.code || 200
//This response is what buildRequestOk is expecting in file hook/backend.ts
if (responseCode >= 200 && responseCode < 300) {
return Promise.resolve({
data: params?.response, config: {
data: params?.response,
params: actualQuery?.params || {},
}, request: { params: actualQuery?.params || {} }
} as any);
}
//This response is what buildRequestFailed is expecting in file hook/backend.ts
return Promise.reject({
response: {
status: responseCode
},
request: {
data: params?.response,
params: actualQuery?.params || {},
}
})
} as any)
this.expectations.push(expectedQuery ? { query: expectedQuery, params, result } : undefined)
}
getLastTestValues(): TestValues {
const expectedQuery = this.expectations.shift()
return [
expectedQuery?.result.args, expectedQuery
]
}
}
export function assertJustExpectedRequestWereMade(env: AxiosMockEnvironment): void {
let size = env.expectations.length
while (size-- > 0) {
assertNextRequest(env)
}
assertNoMoreRequestWereMade(env)
}
export function assertNoMoreRequestWereMade(env: AxiosMockEnvironment): void {
const [actualQuery, expectedQuery] = env.getLastTestValues()
expect(actualQuery).toBeUndefined();
expect(expectedQuery).toBeUndefined();
}
export function assertNextRequest(env: AxiosMockEnvironment): void {
const [actualQuery, expectedQuery] = env.getLastTestValues()
if (!actualQuery) {
//expected one query but the tested component didn't execute one
expect(actualQuery).toBe(expectedQuery);
return
}
if (!expectedQuery) {
const errorMessage = 'a query was made to the backend but the test explicitly expected no query';
if (JEST_DEBUG_LOG) {
console.log(errorMessage, actualQuery)
}
throw Error(errorMessage)
}
if ('get' in expectedQuery.query) {
expect(actualQuery.method).toBe('get');
expect(actualQuery.url).toBe(expectedQuery.query.get);
}
if ('post' in expectedQuery.query) {
expect(actualQuery.method).toBe('post');
expect(actualQuery.url).toBe(expectedQuery.query.post);
}
if ('delete' in expectedQuery.query) {
expect(actualQuery.method).toBe('delete');
expect(actualQuery.url).toBe(expectedQuery.query.delete);
}
if ('patch' in expectedQuery.query) {
expect(actualQuery.method).toBe('patch');
expect(actualQuery.url).toBe(expectedQuery.query.patch);
}
if (expectedQuery.params?.request) {
expect(actualQuery.data).toMatchObject(expectedQuery.params.request)
}
if (expectedQuery.params?.qparam) {
expect(actualQuery.params).toMatchObject(expectedQuery.params.qparam)
}
if (expectedQuery.params?.auth) {
expect(actualQuery.headers.Authorization).toBe(expectedQuery.params?.auth)
}
}
////////////////////
// ORDER
////////////////////
export const API_CREATE_ORDER: Query<
MerchantBackend.Orders.PostOrderRequest,
MerchantBackend.Orders.PostOrderResponse
> = {
post: "http://backend/instances/default/private/orders",
};
export const API_GET_ORDER_BY_ID = (
id: string
): Query<
unknown,
MerchantBackend.Orders.MerchantOrderStatusResponse
> => ({
get: `http://backend/instances/default/private/orders/${id}`,
});
export const API_LIST_ORDERS: Query<
unknown,
MerchantBackend.Orders.OrderHistory
> = {
get: "http://backend/instances/default/private/orders",
};
export const API_REFUND_ORDER_BY_ID = (
id: string
): Query<
MerchantBackend.Orders.RefundRequest,
MerchantBackend.Orders.MerchantRefundResponse
> => ({
post: `http://backend/instances/default/private/orders/${id}/refund`,
});
export const API_FORGET_ORDER_BY_ID = (
id: string
): Query<
MerchantBackend.Orders.ForgetRequest,
unknown
> => ({
patch: `http://backend/instances/default/private/orders/${id}/forget`,
});
export const API_DELETE_ORDER = (
id: string
): Query<
MerchantBackend.Orders.ForgetRequest,
unknown
> => ({
delete: `http://backend/instances/default/private/orders/${id}`,
});
////////////////////
// TRANSFER
////////////////////
export const API_LIST_TRANSFERS: Query<
unknown,
MerchantBackend.Transfers.TransferList
> = {
get: "http://backend/instances/default/private/transfers",
};
export const API_INFORM_TRANSFERS: Query<
MerchantBackend.Transfers.TransferInformation,
MerchantBackend.Transfers.MerchantTrackTransferResponse
> = {
post: "http://backend/instances/default/private/transfers",
};
////////////////////
// PRODUCT
////////////////////
export const API_CREATE_PRODUCT: Query<
MerchantBackend.Products.ProductAddDetail,
unknown
> = {
post: "http://backend/instances/default/private/products",
};
export const API_LIST_PRODUCTS: Query<
unknown,
MerchantBackend.Products.InventorySummaryResponse
> = {
get: "http://backend/instances/default/private/products",
};
export const API_GET_PRODUCT_BY_ID = (
id: string
): Query<unknown, MerchantBackend.Products.ProductDetail> => ({
get: `http://backend/instances/default/private/products/${id}`,
});
export const API_UPDATE_PRODUCT_BY_ID = (
id: string
): Query<
MerchantBackend.Products.ProductPatchDetail,
MerchantBackend.Products.InventorySummaryResponse
> => ({
patch: `http://backend/instances/default/private/products/${id}`,
});
export const API_DELETE_PRODUCT = (
id: string
): Query<
unknown, unknown
> => ({
delete: `http://backend/instances/default/private/products/${id}`,
});
////////////////////
// RESERVES
////////////////////
export const API_CREATE_RESERVE: Query<
MerchantBackend.Tips.ReserveCreateRequest,
MerchantBackend.Tips.ReserveCreateConfirmation
> = {
post: "http://backend/instances/default/private/reserves",
};
export const API_LIST_RESERVES: Query<
unknown,
MerchantBackend.Tips.TippingReserveStatus
> = {
get: "http://backend/instances/default/private/reserves",
};
export const API_GET_RESERVE_BY_ID = (
pub: string
): Query<unknown, MerchantBackend.Tips.ReserveDetail> => ({
get: `http://backend/instances/default/private/reserves/${pub}`,
});
export const API_GET_TIP_BY_ID = (
pub: string
): Query<
unknown,
MerchantBackend.Tips.TipDetails
> => ({
get: `http://backend/instances/default/private/tips/${pub}`,
});
export const API_AUTHORIZE_TIP_FOR_RESERVE = (
pub: string
): Query<
MerchantBackend.Tips.TipCreateRequest,
MerchantBackend.Tips.TipCreateConfirmation
> => ({
post: `http://backend/instances/default/private/reserves/${pub}/authorize-tip`,
});
export const API_AUTHORIZE_TIP: Query<
MerchantBackend.Tips.TipCreateRequest,
MerchantBackend.Tips.TipCreateConfirmation
> = ({
post: `http://backend/instances/default/private/tips`,
});
export const API_DELETE_RESERVE = (
id: string
): Query<unknown, unknown> => ({
delete: `http://backend/instances/default/private/reserves/${id}`,
});
////////////////////
// INSTANCE ADMIN
////////////////////
export const API_CREATE_INSTANCE: Query<
MerchantBackend.Instances.InstanceConfigurationMessage,
unknown
> = {
post: "http://backend/management/instances",
};
export const API_GET_INSTANCE_BY_ID = (
id: string
): Query<
unknown,
MerchantBackend.Instances.QueryInstancesResponse
> => ({
get: `http://backend/management/instances/${id}`,
});
export const API_GET_INSTANCE_KYC_BY_ID = (
id: string
): Query<
unknown,
MerchantBackend.Instances.AccountKycRedirects
> => ({
get: `http://backend/management/instances/${id}/kyc`,
});
export const API_LIST_INSTANCES: Query<
unknown,
MerchantBackend.Instances.InstancesResponse
> = {
get: "http://backend/management/instances",
};
export const API_UPDATE_INSTANCE_BY_ID = (
id: string
): Query<
MerchantBackend.Instances.InstanceReconfigurationMessage,
unknown
> => ({
patch: `http://backend/management/instances/${id}`,
});
export const API_UPDATE_INSTANCE_AUTH_BY_ID = (
id: string
): Query<
MerchantBackend.Instances.InstanceAuthConfigurationMessage,
unknown
> => ({
post: `http://backend/management/instances/${id}/auth`,
});
export const API_DELETE_INSTANCE = (
id: string
): Query<unknown, unknown> => ({
delete: `http://backend/management/instances/${id}`,
});
////////////////////
// INSTANCE
////////////////////
export const API_GET_CURRENT_INSTANCE: Query<
unknown,
MerchantBackend.Instances.QueryInstancesResponse
> = ({
get: `http://backend/instances/default/private/`,
});
export const API_GET_CURRENT_INSTANCE_KYC: Query<
unknown,
MerchantBackend.Instances.AccountKycRedirects
> =
({
get: `http://backend/instances/default/private/kyc`,
});
export const API_UPDATE_CURRENT_INSTANCE: Query<
MerchantBackend.Instances.InstanceReconfigurationMessage,
unknown
> = {
patch: `http://backend/instances/default/private/`,
};
export const API_UPDATE_CURRENT_INSTANCE_AUTH: Query<
MerchantBackend.Instances.InstanceAuthConfigurationMessage,
unknown
> = {
post: `http://backend/instances/default/private/auth`,
};
export const API_DELETE_CURRENT_INSTANCE: Query<
unknown,
unknown
> = ({
delete: `http://backend/instances/default/private`,
});