payment-method.e2e-spec.ts 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652
  1. /* eslint-disable @typescript-eslint/no-non-null-assertion */
  2. import {
  3. DefaultLogger,
  4. dummyPaymentHandler,
  5. LanguageCode,
  6. PaymentMethodEligibilityChecker,
  7. } from '@vendure/core';
  8. import {
  9. createErrorResultGuard,
  10. createTestEnvironment,
  11. E2E_DEFAULT_CHANNEL_TOKEN,
  12. ErrorResultGuard,
  13. } from '@vendure/testing';
  14. import gql from 'graphql-tag';
  15. import path from 'path';
  16. import { vi } from 'vitest';
  17. import { afterAll, beforeAll, describe, expect, it } from 'vitest';
  18. import { initialData } from '../../../e2e-common/e2e-initial-data';
  19. import { testConfig, TEST_SETUP_TIMEOUT_MS } from '../../../e2e-common/test-config';
  20. import { CurrencyCode, DeletionResult } from './graphql/generated-e2e-admin-types';
  21. import * as Codegen from './graphql/generated-e2e-admin-types';
  22. import { ErrorCode } from './graphql/generated-e2e-shop-types';
  23. import * as CodegenShop from './graphql/generated-e2e-shop-types';
  24. import { CREATE_CHANNEL } from './graphql/shared-definitions';
  25. import { ADD_ITEM_TO_ORDER, ADD_PAYMENT, GET_ELIGIBLE_PAYMENT_METHODS } from './graphql/shop-definitions';
  26. import { proceedToArrangingPayment } from './utils/test-order-utils';
  27. const checkerSpy = vi.fn();
  28. const minPriceChecker = new PaymentMethodEligibilityChecker({
  29. code: 'min-price-checker',
  30. description: [{ languageCode: LanguageCode.en, value: 'Min price checker' }],
  31. args: {
  32. minPrice: {
  33. type: 'int',
  34. },
  35. },
  36. check(ctx, order, args) {
  37. checkerSpy();
  38. if (order.totalWithTax >= args.minPrice) {
  39. return true;
  40. } else {
  41. return 'Order total too low';
  42. }
  43. },
  44. });
  45. describe('PaymentMethod resolver', () => {
  46. const orderGuard: ErrorResultGuard<CodegenShop.TestOrderWithPaymentsFragment> = createErrorResultGuard(
  47. input => !!input.lines,
  48. );
  49. const { server, adminClient, shopClient } = createTestEnvironment({
  50. ...testConfig(),
  51. // logger: new DefaultLogger(),
  52. paymentOptions: {
  53. paymentMethodEligibilityCheckers: [minPriceChecker],
  54. paymentMethodHandlers: [dummyPaymentHandler],
  55. },
  56. });
  57. beforeAll(async () => {
  58. await server.init({
  59. initialData,
  60. productsCsvPath: path.join(__dirname, 'fixtures/e2e-products-minimal.csv'),
  61. customerCount: 2,
  62. });
  63. await adminClient.asSuperAdmin();
  64. }, TEST_SETUP_TIMEOUT_MS);
  65. afterAll(async () => {
  66. await server.destroy();
  67. });
  68. it('create', async () => {
  69. const { createPaymentMethod } = await adminClient.query<
  70. Codegen.CreatePaymentMethodMutation,
  71. Codegen.CreatePaymentMethodMutationVariables
  72. >(CREATE_PAYMENT_METHOD, {
  73. input: {
  74. code: 'no-checks',
  75. enabled: true,
  76. handler: {
  77. code: dummyPaymentHandler.code,
  78. arguments: [{ name: 'automaticSettle', value: 'true' }],
  79. },
  80. translations: [
  81. {
  82. languageCode: LanguageCode.en,
  83. name: 'No Checker',
  84. description: 'This is a test payment method',
  85. },
  86. ],
  87. },
  88. });
  89. expect(createPaymentMethod).toEqual({
  90. id: 'T_1',
  91. name: 'No Checker',
  92. code: 'no-checks',
  93. description: 'This is a test payment method',
  94. enabled: true,
  95. checker: null,
  96. handler: {
  97. args: [
  98. {
  99. name: 'automaticSettle',
  100. value: 'true',
  101. },
  102. ],
  103. code: 'dummy-payment-handler',
  104. },
  105. translations: [
  106. {
  107. description: 'This is a test payment method',
  108. id: 'T_1',
  109. languageCode: 'en',
  110. name: 'No Checker',
  111. },
  112. ],
  113. });
  114. });
  115. it('update', async () => {
  116. const { updatePaymentMethod } = await adminClient.query<
  117. Codegen.UpdatePaymentMethodMutation,
  118. Codegen.UpdatePaymentMethodMutationVariables
  119. >(UPDATE_PAYMENT_METHOD, {
  120. input: {
  121. id: 'T_1',
  122. checker: {
  123. code: minPriceChecker.code,
  124. arguments: [{ name: 'minPrice', value: '0' }],
  125. },
  126. handler: {
  127. code: dummyPaymentHandler.code,
  128. arguments: [{ name: 'automaticSettle', value: 'false' }],
  129. },
  130. translations: [
  131. {
  132. languageCode: LanguageCode.en,
  133. description: 'modified',
  134. },
  135. ],
  136. },
  137. });
  138. expect(updatePaymentMethod).toEqual({
  139. id: 'T_1',
  140. name: 'No Checker',
  141. code: 'no-checks',
  142. description: 'modified',
  143. enabled: true,
  144. checker: {
  145. args: [{ name: 'minPrice', value: '0' }],
  146. code: minPriceChecker.code,
  147. },
  148. handler: {
  149. args: [
  150. {
  151. name: 'automaticSettle',
  152. value: 'false',
  153. },
  154. ],
  155. code: dummyPaymentHandler.code,
  156. },
  157. translations: [
  158. {
  159. description: 'modified',
  160. id: 'T_1',
  161. languageCode: 'en',
  162. name: 'No Checker',
  163. },
  164. ],
  165. });
  166. });
  167. it('unset checker', async () => {
  168. const { updatePaymentMethod } = await adminClient.query<
  169. Codegen.UpdatePaymentMethodMutation,
  170. Codegen.UpdatePaymentMethodMutationVariables
  171. >(UPDATE_PAYMENT_METHOD, {
  172. input: {
  173. id: 'T_1',
  174. checker: null,
  175. },
  176. });
  177. expect(updatePaymentMethod.checker).toEqual(null);
  178. const { paymentMethod } = await adminClient.query<
  179. Codegen.GetPaymentMethodQuery,
  180. Codegen.GetPaymentMethodQueryVariables
  181. >(GET_PAYMENT_METHOD, { id: 'T_1' });
  182. expect(paymentMethod!.checker).toEqual(null);
  183. });
  184. it('paymentMethodEligibilityCheckers', async () => {
  185. const { paymentMethodEligibilityCheckers } =
  186. await adminClient.query<Codegen.GetPaymentMethodCheckersQuery>(GET_PAYMENT_METHOD_CHECKERS);
  187. expect(paymentMethodEligibilityCheckers).toEqual([
  188. {
  189. code: minPriceChecker.code,
  190. args: [{ name: 'minPrice', type: 'int' }],
  191. },
  192. ]);
  193. });
  194. it('paymentMethodHandlers', async () => {
  195. const { paymentMethodHandlers } = await adminClient.query<Codegen.GetPaymentMethodHandlersQuery>(
  196. GET_PAYMENT_METHOD_HANDLERS,
  197. );
  198. expect(paymentMethodHandlers).toEqual([
  199. {
  200. code: dummyPaymentHandler.code,
  201. args: [{ name: 'automaticSettle', type: 'boolean' }],
  202. },
  203. ]);
  204. });
  205. describe('eligibility checks', () => {
  206. beforeAll(async () => {
  207. await adminClient.query<
  208. Codegen.CreatePaymentMethodMutation,
  209. Codegen.CreatePaymentMethodMutationVariables
  210. >(CREATE_PAYMENT_METHOD, {
  211. input: {
  212. code: 'price-check',
  213. enabled: true,
  214. checker: {
  215. code: minPriceChecker.code,
  216. arguments: [{ name: 'minPrice', value: '200000' }],
  217. },
  218. handler: {
  219. code: dummyPaymentHandler.code,
  220. arguments: [{ name: 'automaticSettle', value: 'true' }],
  221. },
  222. translations: [
  223. {
  224. languageCode: LanguageCode.en,
  225. name: 'With Min Price Checker',
  226. description: 'Order total must be more than 2k',
  227. },
  228. ],
  229. },
  230. });
  231. await adminClient.query<
  232. Codegen.CreatePaymentMethodMutation,
  233. Codegen.CreatePaymentMethodMutationVariables
  234. >(CREATE_PAYMENT_METHOD, {
  235. input: {
  236. code: 'disabled-method',
  237. enabled: false,
  238. handler: {
  239. code: dummyPaymentHandler.code,
  240. arguments: [{ name: 'automaticSettle', value: 'true' }],
  241. },
  242. translations: [
  243. {
  244. languageCode: LanguageCode.en,
  245. name: 'Disabled ones',
  246. description: 'This method is disabled',
  247. },
  248. ],
  249. },
  250. });
  251. await shopClient.asUserWithCredentials('hayden.zieme12@hotmail.com', 'test');
  252. await shopClient.query<
  253. CodegenShop.AddItemToOrderMutation,
  254. CodegenShop.AddItemToOrderMutationVariables
  255. >(ADD_ITEM_TO_ORDER, {
  256. productVariantId: 'T_1',
  257. quantity: 1,
  258. });
  259. await proceedToArrangingPayment(shopClient);
  260. });
  261. it('eligiblePaymentMethods', async () => {
  262. const { eligiblePaymentMethods } =
  263. await shopClient.query<CodegenShop.GetEligiblePaymentMethodsQuery>(
  264. GET_ELIGIBLE_PAYMENT_METHODS,
  265. );
  266. expect(eligiblePaymentMethods).toEqual([
  267. {
  268. id: 'T_1',
  269. code: 'no-checks',
  270. isEligible: true,
  271. eligibilityMessage: null,
  272. },
  273. {
  274. id: 'T_2',
  275. code: 'price-check',
  276. isEligible: false,
  277. eligibilityMessage: 'Order total too low',
  278. },
  279. ]);
  280. });
  281. it('addPaymentToOrder does not allow ineligible method', async () => {
  282. checkerSpy.mockClear();
  283. const { addPaymentToOrder } = await shopClient.query<
  284. CodegenShop.AddPaymentToOrderMutation,
  285. CodegenShop.AddPaymentToOrderMutationVariables
  286. >(ADD_PAYMENT, {
  287. input: {
  288. method: 'price-check',
  289. metadata: {},
  290. },
  291. });
  292. orderGuard.assertErrorResult(addPaymentToOrder);
  293. expect(addPaymentToOrder.errorCode).toBe(ErrorCode.INELIGIBLE_PAYMENT_METHOD_ERROR);
  294. expect(addPaymentToOrder.eligibilityCheckerMessage).toBe('Order total too low');
  295. expect(checkerSpy).toHaveBeenCalledTimes(1);
  296. });
  297. });
  298. describe('channels', () => {
  299. const SECOND_CHANNEL_TOKEN = 'SECOND_CHANNEL_TOKEN';
  300. const THIRD_CHANNEL_TOKEN = 'THIRD_CHANNEL_TOKEN';
  301. beforeAll(async () => {
  302. await adminClient.query<Codegen.CreateChannelMutation, Codegen.CreateChannelMutationVariables>(
  303. CREATE_CHANNEL,
  304. {
  305. input: {
  306. code: 'second-channel',
  307. token: SECOND_CHANNEL_TOKEN,
  308. defaultLanguageCode: LanguageCode.en,
  309. currencyCode: CurrencyCode.GBP,
  310. pricesIncludeTax: true,
  311. defaultShippingZoneId: 'T_1',
  312. defaultTaxZoneId: 'T_1',
  313. },
  314. },
  315. );
  316. await adminClient.query<Codegen.CreateChannelMutation, Codegen.CreateChannelMutationVariables>(
  317. CREATE_CHANNEL,
  318. {
  319. input: {
  320. code: 'third-channel',
  321. token: THIRD_CHANNEL_TOKEN,
  322. defaultLanguageCode: LanguageCode.en,
  323. currencyCode: CurrencyCode.GBP,
  324. pricesIncludeTax: true,
  325. defaultShippingZoneId: 'T_1',
  326. defaultTaxZoneId: 'T_1',
  327. },
  328. },
  329. );
  330. });
  331. it('creates a PaymentMethod in channel2', async () => {
  332. adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
  333. const { createPaymentMethod } = await adminClient.query<
  334. Codegen.CreatePaymentMethodMutation,
  335. Codegen.CreatePaymentMethodMutationVariables
  336. >(CREATE_PAYMENT_METHOD, {
  337. input: {
  338. code: 'channel-2-method',
  339. enabled: true,
  340. handler: {
  341. code: dummyPaymentHandler.code,
  342. arguments: [{ name: 'automaticSettle', value: 'true' }],
  343. },
  344. translations: [
  345. {
  346. languageCode: LanguageCode.en,
  347. name: 'Channel 2 method',
  348. description: 'This is a test payment method',
  349. },
  350. ],
  351. },
  352. });
  353. expect(createPaymentMethod.code).toBe('channel-2-method');
  354. });
  355. it('method is listed in channel2', async () => {
  356. adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
  357. const { paymentMethods } = await adminClient.query<Codegen.GetPaymentMethodListQuery>(
  358. GET_PAYMENT_METHOD_LIST,
  359. );
  360. expect(paymentMethods.totalItems).toBe(1);
  361. expect(paymentMethods.items[0].code).toBe('channel-2-method');
  362. });
  363. it('method is not listed in channel3', async () => {
  364. adminClient.setChannelToken(THIRD_CHANNEL_TOKEN);
  365. const { paymentMethods } = await adminClient.query<Codegen.GetPaymentMethodListQuery>(
  366. GET_PAYMENT_METHOD_LIST,
  367. );
  368. expect(paymentMethods.totalItems).toBe(0);
  369. });
  370. it('method is listed in default channel', async () => {
  371. adminClient.setChannelToken(E2E_DEFAULT_CHANNEL_TOKEN);
  372. const { paymentMethods } = await adminClient.query<Codegen.GetPaymentMethodListQuery>(
  373. GET_PAYMENT_METHOD_LIST,
  374. );
  375. expect(paymentMethods.totalItems).toBe(4);
  376. expect(paymentMethods.items.map(i => i.code).sort()).toEqual([
  377. 'channel-2-method',
  378. 'disabled-method',
  379. 'no-checks',
  380. 'price-check',
  381. ]);
  382. });
  383. it('delete from channel', async () => {
  384. adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
  385. const { paymentMethods } = await adminClient.query<Codegen.GetPaymentMethodListQuery>(
  386. GET_PAYMENT_METHOD_LIST,
  387. );
  388. expect(paymentMethods.totalItems).toBe(1);
  389. const { deletePaymentMethod } = await adminClient.query<
  390. Codegen.DeletePaymentMethodMutation,
  391. Codegen.DeletePaymentMethodMutationVariables
  392. >(DELETE_PAYMENT_METHOD, {
  393. id: paymentMethods.items[0].id,
  394. });
  395. expect(deletePaymentMethod.result).toBe(DeletionResult.DELETED);
  396. const { paymentMethods: checkChannel } =
  397. await adminClient.query<Codegen.GetPaymentMethodListQuery>(GET_PAYMENT_METHOD_LIST);
  398. expect(checkChannel.totalItems).toBe(0);
  399. adminClient.setChannelToken(E2E_DEFAULT_CHANNEL_TOKEN);
  400. const { paymentMethods: checkDefault } =
  401. await adminClient.query<Codegen.GetPaymentMethodListQuery>(GET_PAYMENT_METHOD_LIST);
  402. expect(checkDefault.totalItems).toBe(4);
  403. });
  404. it('delete from default channel', async () => {
  405. adminClient.setChannelToken(SECOND_CHANNEL_TOKEN);
  406. const { createPaymentMethod } = await adminClient.query<
  407. Codegen.CreatePaymentMethodMutation,
  408. Codegen.CreatePaymentMethodMutationVariables
  409. >(CREATE_PAYMENT_METHOD, {
  410. input: {
  411. code: 'channel-2-method2',
  412. enabled: true,
  413. handler: {
  414. code: dummyPaymentHandler.code,
  415. arguments: [{ name: 'automaticSettle', value: 'true' }],
  416. },
  417. translations: [
  418. {
  419. languageCode: LanguageCode.en,
  420. name: 'Channel 2 method 2',
  421. description: 'This is a test payment method',
  422. },
  423. ],
  424. },
  425. });
  426. adminClient.setChannelToken(E2E_DEFAULT_CHANNEL_TOKEN);
  427. const { deletePaymentMethod: delete1 } = await adminClient.query<
  428. Codegen.DeletePaymentMethodMutation,
  429. Codegen.DeletePaymentMethodMutationVariables
  430. >(DELETE_PAYMENT_METHOD, {
  431. id: createPaymentMethod.id,
  432. });
  433. expect(delete1.result).toBe(DeletionResult.NOT_DELETED);
  434. expect(delete1.message).toBe(
  435. 'The selected PaymentMethod is assigned to the following Channels: second-channel. Set "force: true" to delete from all Channels.',
  436. );
  437. const { paymentMethods: check1 } = await adminClient.query<Codegen.GetPaymentMethodListQuery>(
  438. GET_PAYMENT_METHOD_LIST,
  439. );
  440. expect(check1.totalItems).toBe(5);
  441. const { deletePaymentMethod: delete2 } = await adminClient.query<
  442. Codegen.DeletePaymentMethodMutation,
  443. Codegen.DeletePaymentMethodMutationVariables
  444. >(DELETE_PAYMENT_METHOD, {
  445. id: createPaymentMethod.id,
  446. force: true,
  447. });
  448. expect(delete2.result).toBe(DeletionResult.DELETED);
  449. const { paymentMethods: check2 } = await adminClient.query<Codegen.GetPaymentMethodListQuery>(
  450. GET_PAYMENT_METHOD_LIST,
  451. );
  452. expect(check2.totalItems).toBe(4);
  453. });
  454. });
  455. it('create without description', async () => {
  456. const { createPaymentMethod } = await adminClient.query<
  457. Codegen.CreatePaymentMethodMutation,
  458. Codegen.CreatePaymentMethodMutationVariables
  459. >(CREATE_PAYMENT_METHOD, {
  460. input: {
  461. code: 'no-description',
  462. enabled: true,
  463. handler: {
  464. code: dummyPaymentHandler.code,
  465. arguments: [{ name: 'automaticSettle', value: 'true' }],
  466. },
  467. translations: [
  468. {
  469. languageCode: LanguageCode.en,
  470. name: 'No Description',
  471. },
  472. ],
  473. },
  474. });
  475. expect(createPaymentMethod).toEqual({
  476. id: 'T_6',
  477. name: 'No Description',
  478. code: 'no-description',
  479. description: '',
  480. enabled: true,
  481. checker: null,
  482. handler: {
  483. args: [
  484. {
  485. name: 'automaticSettle',
  486. value: 'true',
  487. },
  488. ],
  489. code: 'dummy-payment-handler',
  490. },
  491. translations: [
  492. {
  493. description: '',
  494. id: 'T_6',
  495. languageCode: 'en',
  496. name: 'No Description',
  497. },
  498. ],
  499. });
  500. });
  501. });
  502. export const PAYMENT_METHOD_FRAGMENT = gql`
  503. fragment PaymentMethod on PaymentMethod {
  504. id
  505. code
  506. name
  507. description
  508. enabled
  509. checker {
  510. code
  511. args {
  512. name
  513. value
  514. }
  515. }
  516. handler {
  517. code
  518. args {
  519. name
  520. value
  521. }
  522. }
  523. translations {
  524. id
  525. languageCode
  526. name
  527. description
  528. }
  529. }
  530. `;
  531. export const CREATE_PAYMENT_METHOD = gql`
  532. mutation CreatePaymentMethod($input: CreatePaymentMethodInput!) {
  533. createPaymentMethod(input: $input) {
  534. ...PaymentMethod
  535. }
  536. }
  537. ${PAYMENT_METHOD_FRAGMENT}
  538. `;
  539. export const UPDATE_PAYMENT_METHOD = gql`
  540. mutation UpdatePaymentMethod($input: UpdatePaymentMethodInput!) {
  541. updatePaymentMethod(input: $input) {
  542. ...PaymentMethod
  543. }
  544. }
  545. ${PAYMENT_METHOD_FRAGMENT}
  546. `;
  547. export const GET_PAYMENT_METHOD_HANDLERS = gql`
  548. query GetPaymentMethodHandlers {
  549. paymentMethodHandlers {
  550. code
  551. args {
  552. name
  553. type
  554. }
  555. }
  556. }
  557. `;
  558. export const GET_PAYMENT_METHOD_CHECKERS = gql`
  559. query GetPaymentMethodCheckers {
  560. paymentMethodEligibilityCheckers {
  561. code
  562. args {
  563. name
  564. type
  565. }
  566. }
  567. }
  568. `;
  569. export const GET_PAYMENT_METHOD = gql`
  570. query GetPaymentMethod($id: ID!) {
  571. paymentMethod(id: $id) {
  572. ...PaymentMethod
  573. }
  574. }
  575. ${PAYMENT_METHOD_FRAGMENT}
  576. `;
  577. export const GET_PAYMENT_METHOD_LIST = gql`
  578. query GetPaymentMethodList($options: PaymentMethodListOptions) {
  579. paymentMethods(options: $options) {
  580. items {
  581. ...PaymentMethod
  582. }
  583. totalItems
  584. }
  585. }
  586. ${PAYMENT_METHOD_FRAGMENT}
  587. `;
  588. export const DELETE_PAYMENT_METHOD = gql`
  589. mutation DeletePaymentMethod($id: ID!, $force: Boolean) {
  590. deletePaymentMethod(id: $id, force: $force) {
  591. message
  592. result
  593. }
  594. }
  595. `;