| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557 |
- import {
- DefaultLogger,
- dummyPaymentHandler,
- LanguageCode,
- PaymentMethodEligibilityChecker,
- } from '@vendure/core';
- import {
- createErrorResultGuard,
- createTestEnvironment,
- E2E_DEFAULT_CHANNEL_TOKEN,
- ErrorResultGuard,
- } from '@vendure/testing';
- import gql from 'graphql-tag';
- import path from 'path';
- import { initialData } from '../../../e2e-common/e2e-initial-data';
- import { testConfig, TEST_SETUP_TIMEOUT_MS } from '../../../e2e-common/test-config';
- import {
- CreateChannel,
- CreatePaymentMethod,
- CurrencyCode,
- DeletePaymentMethod,
- DeletionResult,
- GetPaymentMethod,
- GetPaymentMethodCheckers,
- GetPaymentMethodHandlers,
- GetPaymentMethodList,
- UpdatePaymentMethod,
- } from './graphql/generated-e2e-admin-types';
- import {
- AddItemToOrder,
- AddPaymentToOrder,
- ErrorCode,
- GetEligiblePaymentMethods,
- TestOrderWithPaymentsFragment,
- } from './graphql/generated-e2e-shop-types';
- import { CREATE_CHANNEL } from './graphql/shared-definitions';
- import { ADD_ITEM_TO_ORDER, ADD_PAYMENT, GET_ELIGIBLE_PAYMENT_METHODS } from './graphql/shop-definitions';
- import { proceedToArrangingPayment } from './utils/test-order-utils';
- const checkerSpy = jest.fn();
- const minPriceChecker = new PaymentMethodEligibilityChecker({
- code: 'min-price-checker',
- description: [{ languageCode: LanguageCode.en, value: 'Min price checker' }],
- args: {
- minPrice: {
- type: 'int',
- },
- },
- check(ctx, order, args) {
- checkerSpy();
- if (order.totalWithTax >= args.minPrice) {
- return true;
- } else {
- return `Order total too low`;
- }
- },
- });
- describe('PaymentMethod resolver', () => {
- const orderGuard: ErrorResultGuard<TestOrderWithPaymentsFragment> = createErrorResultGuard(
- input => !!input.lines,
- );
- const { server, adminClient, shopClient } = createTestEnvironment({
- ...testConfig(),
- logger: new DefaultLogger(),
- paymentOptions: {
- paymentMethodEligibilityCheckers: [minPriceChecker],
- paymentMethodHandlers: [dummyPaymentHandler],
- },
- });
- beforeAll(async () => {
- await server.init({
- initialData,
- productsCsvPath: path.join(__dirname, 'fixtures/e2e-products-minimal.csv'),
- customerCount: 2,
- });
- await adminClient.asSuperAdmin();
- }, TEST_SETUP_TIMEOUT_MS);
- afterAll(async () => {
- await server.destroy();
- });
- it('create', async () => {
- const { createPaymentMethod } = await adminClient.query<
- CreatePaymentMethod.Mutation,
- CreatePaymentMethod.Variables
- >(CREATE_PAYMENT_METHOD, {
- input: {
- code: 'no-checks',
- name: 'No Checker',
- description: 'This is a test payment method',
- enabled: true,
- handler: {
- code: dummyPaymentHandler.code,
- arguments: [{ name: 'automaticSettle', value: 'true' }],
- },
- },
- });
- expect(createPaymentMethod).toEqual({
- id: 'T_1',
- name: 'No Checker',
- code: 'no-checks',
- description: 'This is a test payment method',
- enabled: true,
- checker: null,
- handler: {
- args: [
- {
- name: 'automaticSettle',
- value: 'true',
- },
- ],
- code: 'dummy-payment-handler',
- },
- });
- });
- it('update', async () => {
- const { updatePaymentMethod } = await adminClient.query<
- UpdatePaymentMethod.Mutation,
- UpdatePaymentMethod.Variables
- >(UPDATE_PAYMENT_METHOD, {
- input: {
- id: 'T_1',
- description: 'modified',
- checker: {
- code: minPriceChecker.code,
- arguments: [{ name: 'minPrice', value: '0' }],
- },
- handler: {
- code: dummyPaymentHandler.code,
- arguments: [{ name: 'automaticSettle', value: 'false' }],
- },
- },
- });
- expect(updatePaymentMethod).toEqual({
- id: 'T_1',
- name: 'No Checker',
- code: 'no-checks',
- description: 'modified',
- enabled: true,
- checker: {
- args: [{ name: 'minPrice', value: '0' }],
- code: minPriceChecker.code,
- },
- handler: {
- args: [
- {
- name: 'automaticSettle',
- value: 'false',
- },
- ],
- code: dummyPaymentHandler.code,
- },
- });
- });
- it('unset checker', async () => {
- const { updatePaymentMethod } = await adminClient.query<
- UpdatePaymentMethod.Mutation,
- UpdatePaymentMethod.Variables
- >(UPDATE_PAYMENT_METHOD, {
- input: {
- id: 'T_1',
- checker: null,
- },
- });
- expect(updatePaymentMethod.checker).toEqual(null);
- const { paymentMethod } = await adminClient.query<GetPaymentMethod.Query, GetPaymentMethod.Variables>(
- GET_PAYMENT_METHOD,
- { id: 'T_1' },
- );
- expect(paymentMethod.checker).toEqual(null);
- });
- it('paymentMethodEligibilityCheckers', async () => {
- const { paymentMethodEligibilityCheckers } = await adminClient.query<GetPaymentMethodCheckers.Query>(
- GET_PAYMENT_METHOD_CHECKERS,
- );
- expect(paymentMethodEligibilityCheckers).toEqual([
- {
- code: minPriceChecker.code,
- args: [{ name: 'minPrice', type: 'int' }],
- },
- ]);
- });
- it('paymentMethodHandlers', async () => {
- const { paymentMethodHandlers } = await adminClient.query<GetPaymentMethodHandlers.Query>(
- GET_PAYMENT_METHOD_HANDLERS,
- );
- expect(paymentMethodHandlers).toEqual([
- {
- code: dummyPaymentHandler.code,
- args: [{ name: 'automaticSettle', type: 'boolean' }],
- },
- ]);
- });
- describe('eligibility checks', () => {
- beforeAll(async () => {
- await adminClient.query<CreatePaymentMethod.Mutation, CreatePaymentMethod.Variables>(
- CREATE_PAYMENT_METHOD,
- {
- input: {
- code: 'price-check',
- name: 'With Min Price Checker',
- description: 'Order total must be more than 2k',
- enabled: true,
- checker: {
- code: minPriceChecker.code,
- arguments: [{ name: 'minPrice', value: '200000' }],
- },
- handler: {
- code: dummyPaymentHandler.code,
- arguments: [{ name: 'automaticSettle', value: 'true' }],
- },
- },
- },
- );
- await adminClient.query<CreatePaymentMethod.Mutation, CreatePaymentMethod.Variables>(
- CREATE_PAYMENT_METHOD,
- {
- input: {
- code: 'disabled-method',
- name: 'Disabled ones',
- description: 'This method is disabled',
- enabled: false,
- handler: {
- code: dummyPaymentHandler.code,
- arguments: [{ name: 'automaticSettle', value: 'true' }],
- },
- },
- },
- );
- await shopClient.asUserWithCredentials('hayden.zieme12@hotmail.com', 'test');
- await shopClient.query<AddItemToOrder.Mutation, AddItemToOrder.Variables>(ADD_ITEM_TO_ORDER, {
- productVariantId: 'T_1',
- quantity: 1,
- });
- await proceedToArrangingPayment(shopClient);
- });
- it('eligiblePaymentMethods', async () => {
- const { eligiblePaymentMethods } = await shopClient.query<GetEligiblePaymentMethods.Query>(
- GET_ELIGIBLE_PAYMENT_METHODS,
- );
- expect(eligiblePaymentMethods).toEqual([
- {
- id: 'T_1',
- code: 'no-checks',
- isEligible: true,
- eligibilityMessage: null,
- },
- {
- id: 'T_2',
- code: 'price-check',
- isEligible: false,
- eligibilityMessage: 'Order total too low',
- },
- ]);
- });
- it('addPaymentToOrder does not allow ineligible method', async () => {
- checkerSpy.mockClear();
- const { addPaymentToOrder } = await shopClient.query<
- AddPaymentToOrder.Mutation,
- AddPaymentToOrder.Variables
- >(ADD_PAYMENT, {
- input: {
- method: 'price-check',
- metadata: {},
- },
- });
- orderGuard.assertErrorResult(addPaymentToOrder);
- expect(addPaymentToOrder.errorCode).toBe(ErrorCode.INELIGIBLE_PAYMENT_METHOD_ERROR);
- expect(addPaymentToOrder.eligibilityCheckerMessage).toBe('Order total too low');
- expect(checkerSpy).toHaveBeenCalledTimes(1);
- });
- });
- describe('channels', () => {
- const SECOND_CHANNEL_TOKEN = 'SECOND_CHANNEL_TOKEN';
- const THIRD_CHANNEL_TOKEN = 'THIRD_CHANNEL_TOKEN';
- beforeAll(async () => {
- await adminClient.query<CreateChannel.Mutation, CreateChannel.Variables>(CREATE_CHANNEL, {
- input: {
- code: 'second-channel',
- token: SECOND_CHANNEL_TOKEN,
- defaultLanguageCode: LanguageCode.en,
- currencyCode: CurrencyCode.GBP,
- pricesIncludeTax: true,
- defaultShippingZoneId: 'T_1',
- defaultTaxZoneId: 'T_1',
- },
- });
- await adminClient.query<CreateChannel.Mutation, CreateChannel.Variables>(CREATE_CHANNEL, {
- input: {
- code: 'third-channel',
- token: THIRD_CHANNEL_TOKEN,
- defaultLanguageCode: LanguageCode.en,
- currencyCode: CurrencyCode.GBP,
- pricesIncludeTax: true,
- defaultShippingZoneId: 'T_1',
- defaultTaxZoneId: 'T_1',
- },
- });
- });
- it('creates a PaymentMethod in channel2', async () => {
- adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
- const { createPaymentMethod } = await adminClient.query<
- CreatePaymentMethod.Mutation,
- CreatePaymentMethod.Variables
- >(CREATE_PAYMENT_METHOD, {
- input: {
- code: 'channel-2-method',
- name: 'Channel 2 method',
- description: 'This is a test payment method',
- enabled: true,
- handler: {
- code: dummyPaymentHandler.code,
- arguments: [{ name: 'automaticSettle', value: 'true' }],
- },
- },
- });
- expect(createPaymentMethod.code).toBe('channel-2-method');
- });
- it('method is listed in channel2', async () => {
- adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
- const { paymentMethods } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(paymentMethods.totalItems).toBe(1);
- expect(paymentMethods.items[0].code).toBe('channel-2-method');
- });
- it('method is not listed in channel3', async () => {
- adminClient.setChannelToken(THIRD_CHANNEL_TOKEN);
- const { paymentMethods } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(paymentMethods.totalItems).toBe(0);
- });
- it('method is listed in default channel', async () => {
- adminClient.setChannelToken(E2E_DEFAULT_CHANNEL_TOKEN);
- const { paymentMethods } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(paymentMethods.totalItems).toBe(4);
- expect(paymentMethods.items.map(i => i.code).sort()).toEqual([
- 'channel-2-method',
- 'disabled-method',
- 'no-checks',
- 'price-check',
- ]);
- });
- it('delete from channel', async () => {
- adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
- const { paymentMethods } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(paymentMethods.totalItems).toBe(1);
- const { deletePaymentMethod } = await adminClient.query<
- DeletePaymentMethod.Mutation,
- DeletePaymentMethod.Variables
- >(DELETE_PAYMENT_METHOD, {
- id: paymentMethods.items[0].id,
- });
- expect(deletePaymentMethod.result).toBe(DeletionResult.DELETED);
- const { paymentMethods: checkChannel } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(checkChannel.totalItems).toBe(0);
- adminClient.setChannelToken(E2E_DEFAULT_CHANNEL_TOKEN);
- const { paymentMethods: checkDefault } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(checkDefault.totalItems).toBe(4);
- });
- it('delete from default channel', async () => {
- adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
- const { createPaymentMethod } = await adminClient.query<
- CreatePaymentMethod.Mutation,
- CreatePaymentMethod.Variables
- >(CREATE_PAYMENT_METHOD, {
- input: {
- code: 'channel-2-method2',
- name: 'Channel 2 method 2',
- description: 'This is a test payment method',
- enabled: true,
- handler: {
- code: dummyPaymentHandler.code,
- arguments: [{ name: 'automaticSettle', value: 'true' }],
- },
- },
- });
- adminClient.setChannelToken(E2E_DEFAULT_CHANNEL_TOKEN);
- const { deletePaymentMethod: delete1 } = await adminClient.query<
- DeletePaymentMethod.Mutation,
- DeletePaymentMethod.Variables
- >(DELETE_PAYMENT_METHOD, {
- id: createPaymentMethod.id,
- });
- expect(delete1.result).toBe(DeletionResult.NOT_DELETED);
- expect(delete1.message).toBe(
- 'The selected PaymentMethod is assigned to the following Channels: second-channel. Set "force: true" to delete from all Channels.',
- );
- const { paymentMethods: check1 } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(check1.totalItems).toBe(5);
- const { deletePaymentMethod: delete2 } = await adminClient.query<
- DeletePaymentMethod.Mutation,
- DeletePaymentMethod.Variables
- >(DELETE_PAYMENT_METHOD, {
- id: createPaymentMethod.id,
- force: true,
- });
- expect(delete2.result).toBe(DeletionResult.DELETED);
- const { paymentMethods: check2 } = await adminClient.query<GetPaymentMethodList.Query>(
- GET_PAYMENT_METHOD_LIST,
- );
- expect(check2.totalItems).toBe(4);
- });
- });
- });
- export const PAYMENT_METHOD_FRAGMENT = gql`
- fragment PaymentMethod on PaymentMethod {
- id
- code
- name
- description
- enabled
- checker {
- code
- args {
- name
- value
- }
- }
- handler {
- code
- args {
- name
- value
- }
- }
- }
- `;
- export const CREATE_PAYMENT_METHOD = gql`
- mutation CreatePaymentMethod($input: CreatePaymentMethodInput!) {
- createPaymentMethod(input: $input) {
- ...PaymentMethod
- }
- }
- ${PAYMENT_METHOD_FRAGMENT}
- `;
- export const UPDATE_PAYMENT_METHOD = gql`
- mutation UpdatePaymentMethod($input: UpdatePaymentMethodInput!) {
- updatePaymentMethod(input: $input) {
- ...PaymentMethod
- }
- }
- ${PAYMENT_METHOD_FRAGMENT}
- `;
- export const GET_PAYMENT_METHOD_HANDLERS = gql`
- query GetPaymentMethodHandlers {
- paymentMethodHandlers {
- code
- args {
- name
- type
- }
- }
- }
- `;
- export const GET_PAYMENT_METHOD_CHECKERS = gql`
- query GetPaymentMethodCheckers {
- paymentMethodEligibilityCheckers {
- code
- args {
- name
- type
- }
- }
- }
- `;
- export const GET_PAYMENT_METHOD = gql`
- query GetPaymentMethod($id: ID!) {
- paymentMethod(id: $id) {
- ...PaymentMethod
- }
- }
- ${PAYMENT_METHOD_FRAGMENT}
- `;
- export const GET_PAYMENT_METHOD_LIST = gql`
- query GetPaymentMethodList($options: PaymentMethodListOptions) {
- paymentMethods(options: $options) {
- items {
- ...PaymentMethod
- }
- totalItems
- }
- }
- ${PAYMENT_METHOD_FRAGMENT}
- `;
- export const DELETE_PAYMENT_METHOD = gql`
- mutation DeletePaymentMethod($id: ID!, $force: Boolean) {
- deletePaymentMethod(id: $id, force: $force) {
- message
- result
- }
- }
- `;
|