payment-method.e2e-spec.ts 19 KB

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