shop-order.e2e-spec.ts 53 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319
  1. /* tslint:disable:no-non-null-assertion */
  2. import { pick } from '@vendure/common/lib/pick';
  3. import { mergeConfig } from '@vendure/core';
  4. import { createErrorResultGuard, createTestEnvironment, ErrorResultGuard } from '@vendure/testing';
  5. import gql from 'graphql-tag';
  6. import path from 'path';
  7. import { initialData } from '../../../e2e-common/e2e-initial-data';
  8. import { TEST_SETUP_TIMEOUT_MS, testConfig } from '../../../e2e-common/test-config';
  9. import {
  10. testErrorPaymentMethod,
  11. testFailingPaymentMethod,
  12. testSuccessfulPaymentMethod,
  13. } from './fixtures/test-payment-methods';
  14. import {
  15. AttemptLogin,
  16. CreateAddressInput,
  17. GetCountryList,
  18. GetCustomer,
  19. GetCustomerList,
  20. UpdateCountry,
  21. } from './graphql/generated-e2e-admin-types';
  22. import {
  23. ActiveOrderCustomer,
  24. ActiveOrderCustomerFragment,
  25. AddItemToOrder,
  26. AddPaymentToOrder,
  27. AdjustItemQuantity,
  28. ErrorCode,
  29. GetActiveOrder,
  30. GetActiveOrderPayments,
  31. GetActiveOrderWithPayments,
  32. GetAvailableCountries,
  33. GetCustomerAddresses,
  34. GetCustomerOrders,
  35. GetNextOrderStates,
  36. GetOrderByCode,
  37. GetShippingMethods,
  38. PaymentDeclinedError,
  39. RemoveAllOrderLines,
  40. RemoveItemFromOrder,
  41. SetBillingAddress,
  42. SetCustomerForOrder,
  43. SetShippingAddress,
  44. SetShippingMethod,
  45. TestOrderFragmentFragment,
  46. TestOrderWithPaymentsFragment,
  47. TransitionToState,
  48. UpdatedOrderFragment,
  49. } from './graphql/generated-e2e-shop-types';
  50. import {
  51. ATTEMPT_LOGIN,
  52. GET_COUNTRY_LIST,
  53. GET_CUSTOMER,
  54. GET_CUSTOMER_LIST,
  55. UPDATE_COUNTRY,
  56. } from './graphql/shared-definitions';
  57. import {
  58. ADD_ITEM_TO_ORDER,
  59. ADD_PAYMENT,
  60. ADJUST_ITEM_QUANTITY,
  61. GET_ACTIVE_ORDER,
  62. GET_ACTIVE_ORDER_ADDRESSES,
  63. GET_ACTIVE_ORDER_ORDERS,
  64. GET_ACTIVE_ORDER_PAYMENTS,
  65. GET_ACTIVE_ORDER_WITH_PAYMENTS,
  66. GET_AVAILABLE_COUNTRIES,
  67. GET_ELIGIBLE_SHIPPING_METHODS,
  68. GET_NEXT_STATES,
  69. GET_ORDER_BY_CODE,
  70. REMOVE_ALL_ORDER_LINES,
  71. REMOVE_ITEM_FROM_ORDER,
  72. SET_BILLING_ADDRESS,
  73. SET_CUSTOMER,
  74. SET_SHIPPING_ADDRESS,
  75. SET_SHIPPING_METHOD,
  76. TEST_ORDER_FRAGMENT,
  77. TRANSITION_TO_STATE,
  78. } from './graphql/shop-definitions';
  79. import { assertThrowsWithMessage } from './utils/assert-throws-with-message';
  80. describe('Shop orders', () => {
  81. const { server, adminClient, shopClient } = createTestEnvironment(
  82. mergeConfig(testConfig, {
  83. paymentOptions: {
  84. paymentMethodHandlers: [
  85. testSuccessfulPaymentMethod,
  86. testFailingPaymentMethod,
  87. testErrorPaymentMethod,
  88. ],
  89. },
  90. customFields: {
  91. Order: [{ name: 'giftWrap', type: 'boolean', defaultValue: false }],
  92. },
  93. orderOptions: {
  94. orderItemsLimit: 99,
  95. },
  96. }),
  97. );
  98. type OrderSuccessResult =
  99. | UpdatedOrderFragment
  100. | TestOrderFragmentFragment
  101. | TestOrderWithPaymentsFragment
  102. | ActiveOrderCustomerFragment;
  103. const orderResultGuard: ErrorResultGuard<OrderSuccessResult> = createErrorResultGuard<OrderSuccessResult>(
  104. input => !!input.lines,
  105. );
  106. beforeAll(async () => {
  107. await server.init({
  108. initialData,
  109. productsCsvPath: path.join(__dirname, 'fixtures/e2e-products-full.csv'),
  110. customerCount: 3,
  111. });
  112. await adminClient.asSuperAdmin();
  113. }, TEST_SETUP_TIMEOUT_MS);
  114. afterAll(async () => {
  115. await server.destroy();
  116. });
  117. it('availableCountries returns enabled countries', async () => {
  118. // disable Austria
  119. const { countries } = await adminClient.query<GetCountryList.Query>(GET_COUNTRY_LIST, {});
  120. const AT = countries.items.find(c => c.code === 'AT')!;
  121. await adminClient.query<UpdateCountry.Mutation, UpdateCountry.Variables>(UPDATE_COUNTRY, {
  122. input: {
  123. id: AT.id,
  124. enabled: false,
  125. },
  126. });
  127. const result = await shopClient.query<GetAvailableCountries.Query>(GET_AVAILABLE_COUNTRIES);
  128. expect(result.availableCountries.length).toBe(countries.items.length - 1);
  129. expect(result.availableCountries.find(c => c.id === AT.id)).toBeUndefined();
  130. });
  131. describe('ordering as anonymous user', () => {
  132. let firstOrderLineId: string;
  133. let createdCustomerId: string;
  134. let orderCode: string;
  135. it('addItemToOrder starts with no session token', () => {
  136. expect(shopClient.getAuthToken()).toBeFalsy();
  137. });
  138. it('activeOrder returns null before any items have been added', async () => {
  139. const result = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  140. expect(result.activeOrder).toBeNull();
  141. });
  142. it('activeOrder creates an anonymous session', () => {
  143. expect(shopClient.getAuthToken()).not.toBe('');
  144. });
  145. it('addItemToOrder creates a new Order with an item', async () => {
  146. const { addItemToOrder } = await shopClient.query<
  147. AddItemToOrder.Mutation,
  148. AddItemToOrder.Variables
  149. >(ADD_ITEM_TO_ORDER, {
  150. productVariantId: 'T_1',
  151. quantity: 1,
  152. });
  153. orderResultGuard.assertSuccess(addItemToOrder);
  154. expect(addItemToOrder!.lines.length).toBe(1);
  155. expect(addItemToOrder!.lines[0].quantity).toBe(1);
  156. expect(addItemToOrder!.lines[0].productVariant.id).toBe('T_1');
  157. expect(addItemToOrder!.lines[0].id).toBe('T_1');
  158. firstOrderLineId = addItemToOrder!.lines[0].id;
  159. orderCode = addItemToOrder!.code;
  160. });
  161. it(
  162. 'addItemToOrder errors with an invalid productVariantId',
  163. assertThrowsWithMessage(
  164. () =>
  165. shopClient.query<AddItemToOrder.Mutation, AddItemToOrder.Variables>(ADD_ITEM_TO_ORDER, {
  166. productVariantId: 'T_999',
  167. quantity: 1,
  168. }),
  169. `No ProductVariant with the id '999' could be found`,
  170. ),
  171. );
  172. it('addItemToOrder errors with a negative quantity', async () => {
  173. const { addItemToOrder } = await shopClient.query<
  174. AddItemToOrder.Mutation,
  175. AddItemToOrder.Variables
  176. >(ADD_ITEM_TO_ORDER, {
  177. productVariantId: 'T_999',
  178. quantity: -3,
  179. });
  180. orderResultGuard.assertErrorResult(addItemToOrder);
  181. expect(addItemToOrder.message).toEqual(`The quantity for an OrderItem cannot be negative`);
  182. expect(addItemToOrder.errorCode).toEqual(ErrorCode.NEGATIVE_QUANTITY_ERROR);
  183. });
  184. it('addItemToOrder with an existing productVariantId adds quantity to the existing OrderLine', async () => {
  185. const { addItemToOrder } = await shopClient.query<
  186. AddItemToOrder.Mutation,
  187. AddItemToOrder.Variables
  188. >(ADD_ITEM_TO_ORDER, {
  189. productVariantId: 'T_1',
  190. quantity: 2,
  191. });
  192. orderResultGuard.assertSuccess(addItemToOrder);
  193. expect(addItemToOrder!.lines.length).toBe(1);
  194. expect(addItemToOrder!.lines[0].quantity).toBe(3);
  195. });
  196. it('addItemToOrder errors when going beyond orderItemsLimit', async () => {
  197. const { addItemToOrder } = await shopClient.query<
  198. AddItemToOrder.Mutation,
  199. AddItemToOrder.Variables
  200. >(ADD_ITEM_TO_ORDER, {
  201. productVariantId: 'T_1',
  202. quantity: 100,
  203. });
  204. orderResultGuard.assertErrorResult(addItemToOrder);
  205. expect(addItemToOrder.message).toBe(
  206. 'Cannot add items. An order may consist of a maximum of 99 items',
  207. );
  208. expect(addItemToOrder.errorCode).toBe(ErrorCode.ORDER_LIMIT_ERROR);
  209. });
  210. it('adjustOrderLine adjusts the quantity', async () => {
  211. const { adjustOrderLine } = await shopClient.query<
  212. AdjustItemQuantity.Mutation,
  213. AdjustItemQuantity.Variables
  214. >(ADJUST_ITEM_QUANTITY, {
  215. orderLineId: firstOrderLineId,
  216. quantity: 50,
  217. });
  218. orderResultGuard.assertSuccess(adjustOrderLine);
  219. expect(adjustOrderLine!.lines.length).toBe(1);
  220. expect(adjustOrderLine!.lines[0].quantity).toBe(50);
  221. });
  222. it('adjustOrderLine with quantity 0 removes the line', async () => {
  223. const { addItemToOrder } = await shopClient.query<
  224. AddItemToOrder.Mutation,
  225. AddItemToOrder.Variables
  226. >(ADD_ITEM_TO_ORDER, {
  227. productVariantId: 'T_3',
  228. quantity: 3,
  229. });
  230. orderResultGuard.assertSuccess(addItemToOrder);
  231. expect(addItemToOrder!.lines.length).toBe(2);
  232. expect(addItemToOrder!.lines.map(i => i.productVariant.id)).toEqual(['T_1', 'T_3']);
  233. const { adjustOrderLine } = await shopClient.query<
  234. AdjustItemQuantity.Mutation,
  235. AdjustItemQuantity.Variables
  236. >(ADJUST_ITEM_QUANTITY, {
  237. orderLineId: addItemToOrder?.lines[1].id!,
  238. quantity: 0,
  239. });
  240. orderResultGuard.assertSuccess(adjustOrderLine);
  241. expect(adjustOrderLine!.lines.length).toBe(1);
  242. expect(adjustOrderLine!.lines.map(i => i.productVariant.id)).toEqual(['T_1']);
  243. });
  244. it('adjustOrderLine errors when going beyond orderItemsLimit', async () => {
  245. const { adjustOrderLine } = await shopClient.query<
  246. AdjustItemQuantity.Mutation,
  247. AdjustItemQuantity.Variables
  248. >(ADJUST_ITEM_QUANTITY, {
  249. orderLineId: firstOrderLineId,
  250. quantity: 100,
  251. });
  252. orderResultGuard.assertErrorResult(adjustOrderLine);
  253. expect(adjustOrderLine.message).toBe(
  254. 'Cannot add items. An order may consist of a maximum of 99 items',
  255. );
  256. expect(adjustOrderLine.errorCode).toBe(ErrorCode.ORDER_LIMIT_ERROR);
  257. });
  258. it('adjustOrderLine errors with a negative quantity', async () => {
  259. const { adjustOrderLine } = await shopClient.query<
  260. AdjustItemQuantity.Mutation,
  261. AdjustItemQuantity.Variables
  262. >(ADJUST_ITEM_QUANTITY, {
  263. orderLineId: firstOrderLineId,
  264. quantity: -3,
  265. });
  266. orderResultGuard.assertErrorResult(adjustOrderLine);
  267. expect(adjustOrderLine.message).toBe('The quantity for an OrderItem cannot be negative');
  268. expect(adjustOrderLine.errorCode).toBe(ErrorCode.NEGATIVE_QUANTITY_ERROR);
  269. });
  270. it(
  271. 'adjustOrderLine errors with an invalid orderLineId',
  272. assertThrowsWithMessage(
  273. () =>
  274. shopClient.query<AdjustItemQuantity.Mutation, AdjustItemQuantity.Variables>(
  275. ADJUST_ITEM_QUANTITY,
  276. {
  277. orderLineId: 'T_999',
  278. quantity: 5,
  279. },
  280. ),
  281. `This order does not contain an OrderLine with the id 999`,
  282. ),
  283. );
  284. it('removeItemFromOrder removes the correct item', async () => {
  285. const { addItemToOrder } = await shopClient.query<
  286. AddItemToOrder.Mutation,
  287. AddItemToOrder.Variables
  288. >(ADD_ITEM_TO_ORDER, {
  289. productVariantId: 'T_3',
  290. quantity: 3,
  291. });
  292. orderResultGuard.assertSuccess(addItemToOrder);
  293. expect(addItemToOrder!.lines.length).toBe(2);
  294. expect(addItemToOrder!.lines.map(i => i.productVariant.id)).toEqual(['T_1', 'T_3']);
  295. const { removeOrderLine } = await shopClient.query<
  296. RemoveItemFromOrder.Mutation,
  297. RemoveItemFromOrder.Variables
  298. >(REMOVE_ITEM_FROM_ORDER, {
  299. orderLineId: firstOrderLineId,
  300. });
  301. orderResultGuard.assertSuccess(removeOrderLine);
  302. expect(removeOrderLine!.lines.length).toBe(1);
  303. expect(removeOrderLine!.lines.map(i => i.productVariant.id)).toEqual(['T_3']);
  304. });
  305. it(
  306. 'removeItemFromOrder errors with an invalid orderItemId',
  307. assertThrowsWithMessage(
  308. () =>
  309. shopClient.query<RemoveItemFromOrder.Mutation, RemoveItemFromOrder.Variables>(
  310. REMOVE_ITEM_FROM_ORDER,
  311. {
  312. orderLineId: 'T_999',
  313. },
  314. ),
  315. `This order does not contain an OrderLine with the id 999`,
  316. ),
  317. );
  318. it('nextOrderStates returns next valid states', async () => {
  319. const result = await shopClient.query<GetNextOrderStates.Query>(GET_NEXT_STATES);
  320. expect(result.nextOrderStates).toEqual(['ArrangingPayment', 'Cancelled']);
  321. });
  322. it('transitionOrderToState returns error result for invalid state', async () => {
  323. const { transitionOrderToState } = await shopClient.query<
  324. TransitionToState.Mutation,
  325. TransitionToState.Variables
  326. >(TRANSITION_TO_STATE, { state: 'Completed' });
  327. orderResultGuard.assertErrorResult(transitionOrderToState);
  328. expect(transitionOrderToState!.message).toBe(
  329. `Cannot transition Order from "AddingItems" to "Completed"`,
  330. );
  331. expect(transitionOrderToState!.errorCode).toBe(ErrorCode.ORDER_STATE_TRANSITION_ERROR);
  332. });
  333. it('attempting to transition to ArrangingPayment returns error result when Order has no Customer', async () => {
  334. const { transitionOrderToState } = await shopClient.query<
  335. TransitionToState.Mutation,
  336. TransitionToState.Variables
  337. >(TRANSITION_TO_STATE, { state: 'ArrangingPayment' });
  338. orderResultGuard.assertErrorResult(transitionOrderToState);
  339. expect(transitionOrderToState!.transitionError).toBe(
  340. `Cannot transition Order to the "ArrangingPayment" state without Customer details`,
  341. );
  342. expect(transitionOrderToState!.errorCode).toBe(ErrorCode.ORDER_STATE_TRANSITION_ERROR);
  343. });
  344. it('setCustomerForOrder returns error result on email address conflict', async () => {
  345. const { customers } = await adminClient.query<GetCustomerList.Query>(GET_CUSTOMER_LIST);
  346. const { setCustomerForOrder } = await shopClient.query<
  347. SetCustomerForOrder.Mutation,
  348. SetCustomerForOrder.Variables
  349. >(SET_CUSTOMER, {
  350. input: {
  351. emailAddress: customers.items[0].emailAddress,
  352. firstName: 'Test',
  353. lastName: 'Person',
  354. },
  355. });
  356. orderResultGuard.assertErrorResult(setCustomerForOrder);
  357. expect(setCustomerForOrder!.message).toBe('The email address is not available.');
  358. expect(setCustomerForOrder!.errorCode).toBe(ErrorCode.EMAIL_ADDRESS_CONFLICT_ERROR);
  359. });
  360. it('setCustomerForOrder creates a new Customer and associates it with the Order', async () => {
  361. const { setCustomerForOrder } = await shopClient.query<
  362. SetCustomerForOrder.Mutation,
  363. SetCustomerForOrder.Variables
  364. >(SET_CUSTOMER, {
  365. input: {
  366. emailAddress: 'test@test.com',
  367. firstName: 'Test',
  368. lastName: 'Person',
  369. },
  370. });
  371. orderResultGuard.assertSuccess(setCustomerForOrder);
  372. const customer = setCustomerForOrder!.customer!;
  373. expect(customer.firstName).toBe('Test');
  374. expect(customer.lastName).toBe('Person');
  375. expect(customer.emailAddress).toBe('test@test.com');
  376. createdCustomerId = customer.id;
  377. });
  378. it('setCustomerForOrder updates the existing customer if Customer already set', async () => {
  379. const { setCustomerForOrder } = await shopClient.query<
  380. SetCustomerForOrder.Mutation,
  381. SetCustomerForOrder.Variables
  382. >(SET_CUSTOMER, {
  383. input: {
  384. emailAddress: 'test@test.com',
  385. firstName: 'Changed',
  386. lastName: 'Person',
  387. },
  388. });
  389. orderResultGuard.assertSuccess(setCustomerForOrder);
  390. const customer = setCustomerForOrder!.customer!;
  391. expect(customer.firstName).toBe('Changed');
  392. expect(customer.lastName).toBe('Person');
  393. expect(customer.emailAddress).toBe('test@test.com');
  394. expect(customer.id).toBe(createdCustomerId);
  395. });
  396. it('setOrderShippingAddress sets shipping address', async () => {
  397. const address: CreateAddressInput = {
  398. fullName: 'name',
  399. company: 'company',
  400. streetLine1: '12 the street',
  401. streetLine2: null,
  402. city: 'foo',
  403. province: 'bar',
  404. postalCode: '123456',
  405. countryCode: 'US',
  406. phoneNumber: '4444444',
  407. };
  408. const { setOrderShippingAddress } = await shopClient.query<
  409. SetShippingAddress.Mutation,
  410. SetShippingAddress.Variables
  411. >(SET_SHIPPING_ADDRESS, {
  412. input: address,
  413. });
  414. expect(setOrderShippingAddress!.shippingAddress).toEqual({
  415. fullName: 'name',
  416. company: 'company',
  417. streetLine1: '12 the street',
  418. streetLine2: null,
  419. city: 'foo',
  420. province: 'bar',
  421. postalCode: '123456',
  422. country: 'United States of America',
  423. phoneNumber: '4444444',
  424. });
  425. });
  426. it('setOrderBillingAddress sets billing address', async () => {
  427. const address: CreateAddressInput = {
  428. fullName: 'name',
  429. company: 'company',
  430. streetLine1: '12 the street',
  431. streetLine2: null,
  432. city: 'foo',
  433. province: 'bar',
  434. postalCode: '123456',
  435. countryCode: 'US',
  436. phoneNumber: '4444444',
  437. };
  438. const { setOrderBillingAddress } = await shopClient.query<
  439. SetBillingAddress.Mutation,
  440. SetBillingAddress.Variables
  441. >(SET_BILLING_ADDRESS, {
  442. input: address,
  443. });
  444. expect(setOrderBillingAddress!.billingAddress).toEqual({
  445. fullName: 'name',
  446. company: 'company',
  447. streetLine1: '12 the street',
  448. streetLine2: null,
  449. city: 'foo',
  450. province: 'bar',
  451. postalCode: '123456',
  452. country: 'United States of America',
  453. phoneNumber: '4444444',
  454. });
  455. });
  456. it('customer default Addresses are not updated before payment', async () => {
  457. const { activeOrder } = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  458. const { customer } = await adminClient.query<GetCustomer.Query, GetCustomer.Variables>(
  459. GET_CUSTOMER,
  460. { id: activeOrder!.customer!.id },
  461. );
  462. expect(customer!.addresses).toEqual([]);
  463. });
  464. it('can transition to ArrangingPayment once Customer has been set', async () => {
  465. const { transitionOrderToState } = await shopClient.query<
  466. TransitionToState.Mutation,
  467. TransitionToState.Variables
  468. >(TRANSITION_TO_STATE, { state: 'ArrangingPayment' });
  469. orderResultGuard.assertSuccess(transitionOrderToState);
  470. expect(pick(transitionOrderToState, ['id', 'state'])).toEqual({
  471. id: 'T_1',
  472. state: 'ArrangingPayment',
  473. });
  474. });
  475. it('adds a successful payment and transitions Order state', async () => {
  476. const { addPaymentToOrder } = await shopClient.query<
  477. AddPaymentToOrder.Mutation,
  478. AddPaymentToOrder.Variables
  479. >(ADD_PAYMENT, {
  480. input: {
  481. method: testSuccessfulPaymentMethod.code,
  482. metadata: {},
  483. },
  484. });
  485. orderResultGuard.assertSuccess(addPaymentToOrder);
  486. const payment = addPaymentToOrder!.payments![0];
  487. expect(addPaymentToOrder!.state).toBe('PaymentSettled');
  488. expect(addPaymentToOrder!.active).toBe(false);
  489. expect(addPaymentToOrder!.payments!.length).toBe(1);
  490. expect(payment.method).toBe(testSuccessfulPaymentMethod.code);
  491. expect(payment.state).toBe('Settled');
  492. });
  493. it('activeOrder is null after payment', async () => {
  494. const result = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  495. expect(result.activeOrder).toBeNull();
  496. });
  497. it('customer default Addresses are updated after payment', async () => {
  498. const result = await adminClient.query<GetCustomer.Query, GetCustomer.Variables>(GET_CUSTOMER, {
  499. id: createdCustomerId,
  500. });
  501. // tslint:disable-next-line:no-non-null-assertion
  502. const address = result.customer!.addresses![0];
  503. expect(address.streetLine1).toBe('12 the street');
  504. expect(address.postalCode).toBe('123456');
  505. expect(address.defaultBillingAddress).toBe(true);
  506. expect(address.defaultShippingAddress).toBe(true);
  507. });
  508. });
  509. describe('ordering as authenticated user', () => {
  510. let firstOrderLineId: string;
  511. let activeOrder: UpdatedOrderFragment;
  512. let authenticatedUserEmailAddress: string;
  513. let customers: GetCustomerList.Items[];
  514. const password = 'test';
  515. beforeAll(async () => {
  516. await adminClient.asSuperAdmin();
  517. const result = await adminClient.query<GetCustomerList.Query, GetCustomerList.Variables>(
  518. GET_CUSTOMER_LIST,
  519. {
  520. options: {
  521. take: 2,
  522. },
  523. },
  524. );
  525. customers = result.customers.items;
  526. authenticatedUserEmailAddress = customers[0].emailAddress;
  527. await shopClient.asUserWithCredentials(authenticatedUserEmailAddress, password);
  528. });
  529. it('activeOrder returns null before any items have been added', async () => {
  530. const result = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  531. expect(result.activeOrder).toBeNull();
  532. });
  533. it('addItemToOrder creates a new Order with an item', async () => {
  534. const { addItemToOrder } = await shopClient.query<
  535. AddItemToOrder.Mutation,
  536. AddItemToOrder.Variables
  537. >(ADD_ITEM_TO_ORDER, {
  538. productVariantId: 'T_1',
  539. quantity: 1,
  540. });
  541. orderResultGuard.assertSuccess(addItemToOrder);
  542. expect(addItemToOrder!.lines.length).toBe(1);
  543. expect(addItemToOrder!.lines[0].quantity).toBe(1);
  544. expect(addItemToOrder!.lines[0].productVariant.id).toBe('T_1');
  545. activeOrder = addItemToOrder!;
  546. firstOrderLineId = addItemToOrder!.lines[0].id;
  547. });
  548. it('activeOrder returns order after item has been added', async () => {
  549. const result = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  550. expect(result.activeOrder!.id).toBe(activeOrder.id);
  551. expect(result.activeOrder!.state).toBe('AddingItems');
  552. });
  553. it('activeOrder resolves customer user', async () => {
  554. const result = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  555. expect(result.activeOrder!.customer!.user).toEqual({
  556. id: 'T_2',
  557. identifier: 'hayden.zieme12@hotmail.com',
  558. });
  559. });
  560. it('addItemToOrder with an existing productVariantId adds quantity to the existing OrderLine', async () => {
  561. const { addItemToOrder } = await shopClient.query<
  562. AddItemToOrder.Mutation,
  563. AddItemToOrder.Variables
  564. >(ADD_ITEM_TO_ORDER, {
  565. productVariantId: 'T_1',
  566. quantity: 2,
  567. });
  568. orderResultGuard.assertSuccess(addItemToOrder);
  569. expect(addItemToOrder!.lines.length).toBe(1);
  570. expect(addItemToOrder!.lines[0].quantity).toBe(3);
  571. });
  572. it('adjustOrderLine adjusts the quantity', async () => {
  573. const { adjustOrderLine } = await shopClient.query<
  574. AdjustItemQuantity.Mutation,
  575. AdjustItemQuantity.Variables
  576. >(ADJUST_ITEM_QUANTITY, {
  577. orderLineId: firstOrderLineId,
  578. quantity: 50,
  579. });
  580. orderResultGuard.assertSuccess(adjustOrderLine);
  581. expect(adjustOrderLine!.lines.length).toBe(1);
  582. expect(adjustOrderLine!.lines[0].quantity).toBe(50);
  583. });
  584. it('removeItemFromOrder removes the correct item', async () => {
  585. const { addItemToOrder } = await shopClient.query<
  586. AddItemToOrder.Mutation,
  587. AddItemToOrder.Variables
  588. >(ADD_ITEM_TO_ORDER, {
  589. productVariantId: 'T_3',
  590. quantity: 3,
  591. });
  592. orderResultGuard.assertSuccess(addItemToOrder);
  593. expect(addItemToOrder!.lines.length).toBe(2);
  594. expect(addItemToOrder!.lines.map(i => i.productVariant.id)).toEqual(['T_1', 'T_3']);
  595. const { removeOrderLine } = await shopClient.query<
  596. RemoveItemFromOrder.Mutation,
  597. RemoveItemFromOrder.Variables
  598. >(REMOVE_ITEM_FROM_ORDER, {
  599. orderLineId: firstOrderLineId,
  600. });
  601. orderResultGuard.assertSuccess(removeOrderLine);
  602. expect(removeOrderLine!.lines.length).toBe(1);
  603. expect(removeOrderLine!.lines.map(i => i.productVariant.id)).toEqual(['T_3']);
  604. });
  605. it('nextOrderStates returns next valid states', async () => {
  606. const result = await shopClient.query<GetNextOrderStates.Query>(GET_NEXT_STATES);
  607. expect(result.nextOrderStates).toEqual(['ArrangingPayment', 'Cancelled']);
  608. });
  609. it('logging out and back in again resumes the last active order', async () => {
  610. await shopClient.asAnonymousUser();
  611. const result1 = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  612. expect(result1.activeOrder).toBeNull();
  613. await shopClient.asUserWithCredentials(authenticatedUserEmailAddress, password);
  614. const result2 = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  615. expect(result2.activeOrder!.id).toBe(activeOrder.id);
  616. });
  617. it('cannot setCustomerForOrder when already logged in', async () => {
  618. const { setCustomerForOrder } = await shopClient.query<
  619. SetCustomerForOrder.Mutation,
  620. SetCustomerForOrder.Variables
  621. >(SET_CUSTOMER, {
  622. input: {
  623. emailAddress: 'newperson@email.com',
  624. firstName: 'New',
  625. lastName: 'Person',
  626. },
  627. });
  628. orderResultGuard.assertErrorResult(setCustomerForOrder);
  629. expect(setCustomerForOrder!.message).toBe(
  630. 'Cannot set a Customer for the Order when already logged in',
  631. );
  632. expect(setCustomerForOrder!.errorCode).toBe(ErrorCode.ALREADY_LOGGED_IN_ERROR);
  633. });
  634. describe('shipping', () => {
  635. let shippingMethods: GetShippingMethods.EligibleShippingMethods[];
  636. it(
  637. 'setOrderShippingAddress throws with invalid countryCode',
  638. assertThrowsWithMessage(() => {
  639. const address: CreateAddressInput = {
  640. streetLine1: '12 the street',
  641. countryCode: 'INVALID',
  642. };
  643. return shopClient.query<SetShippingAddress.Mutation, SetShippingAddress.Variables>(
  644. SET_SHIPPING_ADDRESS,
  645. {
  646. input: address,
  647. },
  648. );
  649. }, `The countryCode "INVALID" was not recognized`),
  650. );
  651. it('setOrderShippingAddress sets shipping address', async () => {
  652. const address: CreateAddressInput = {
  653. fullName: 'name',
  654. company: 'company',
  655. streetLine1: '12 the street',
  656. streetLine2: null,
  657. city: 'foo',
  658. province: 'bar',
  659. postalCode: '123456',
  660. countryCode: 'US',
  661. phoneNumber: '4444444',
  662. };
  663. const { setOrderShippingAddress } = await shopClient.query<
  664. SetShippingAddress.Mutation,
  665. SetShippingAddress.Variables
  666. >(SET_SHIPPING_ADDRESS, {
  667. input: address,
  668. });
  669. expect(setOrderShippingAddress!.shippingAddress).toEqual({
  670. fullName: 'name',
  671. company: 'company',
  672. streetLine1: '12 the street',
  673. streetLine2: null,
  674. city: 'foo',
  675. province: 'bar',
  676. postalCode: '123456',
  677. country: 'United States of America',
  678. phoneNumber: '4444444',
  679. });
  680. });
  681. it('eligibleShippingMethods lists shipping methods', async () => {
  682. const result = await shopClient.query<GetShippingMethods.Query>(
  683. GET_ELIGIBLE_SHIPPING_METHODS,
  684. );
  685. shippingMethods = result.eligibleShippingMethods;
  686. expect(shippingMethods).toEqual([
  687. { id: 'T_1', price: 500, description: 'Standard Shipping' },
  688. { id: 'T_2', price: 1000, description: 'Express Shipping' },
  689. ]);
  690. });
  691. it('shipping is initially unset', async () => {
  692. const result = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  693. expect(result.activeOrder!.shipping).toEqual(0);
  694. expect(result.activeOrder!.shippingMethod).toEqual(null);
  695. });
  696. it('setOrderShippingMethod sets the shipping method', async () => {
  697. const result = await shopClient.query<
  698. SetShippingMethod.Mutation,
  699. SetShippingMethod.Variables
  700. >(SET_SHIPPING_METHOD, {
  701. id: shippingMethods[1].id,
  702. });
  703. const activeOrderResult = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  704. const order = activeOrderResult.activeOrder!;
  705. expect(order.shipping).toBe(shippingMethods[1].price);
  706. expect(order.shippingMethod!.id).toBe(shippingMethods[1].id);
  707. expect(order.shippingMethod!.description).toBe(shippingMethods[1].description);
  708. });
  709. it('shipping method is preserved after adjustOrderLine', async () => {
  710. const activeOrderResult = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  711. activeOrder = activeOrderResult.activeOrder!;
  712. const { adjustOrderLine } = await shopClient.query<
  713. AdjustItemQuantity.Mutation,
  714. AdjustItemQuantity.Variables
  715. >(ADJUST_ITEM_QUANTITY, {
  716. orderLineId: activeOrder.lines[0].id,
  717. quantity: 10,
  718. });
  719. orderResultGuard.assertSuccess(adjustOrderLine);
  720. expect(adjustOrderLine!.shipping).toBe(shippingMethods[1].price);
  721. expect(adjustOrderLine!.shippingMethod!.id).toBe(shippingMethods[1].id);
  722. expect(adjustOrderLine!.shippingMethod!.description).toBe(shippingMethods[1].description);
  723. });
  724. });
  725. describe('payment', () => {
  726. it('attempting add a Payment returns error result when in AddingItems state', async () => {
  727. const { addPaymentToOrder } = await shopClient.query<
  728. AddPaymentToOrder.Mutation,
  729. AddPaymentToOrder.Variables
  730. >(ADD_PAYMENT, {
  731. input: {
  732. method: testSuccessfulPaymentMethod.code,
  733. metadata: {},
  734. },
  735. });
  736. orderResultGuard.assertErrorResult(addPaymentToOrder);
  737. expect(addPaymentToOrder!.message).toBe(
  738. `A Payment may only be added when Order is in "ArrangingPayment" state`,
  739. );
  740. expect(addPaymentToOrder!.errorCode).toBe(ErrorCode.ORDER_PAYMENT_STATE_ERROR);
  741. });
  742. it('transitions to the ArrangingPayment state', async () => {
  743. const { transitionOrderToState } = await shopClient.query<
  744. TransitionToState.Mutation,
  745. TransitionToState.Variables
  746. >(TRANSITION_TO_STATE, { state: 'ArrangingPayment' });
  747. orderResultGuard.assertSuccess(transitionOrderToState);
  748. expect(pick(transitionOrderToState, ['id', 'state'])).toEqual({
  749. id: activeOrder.id,
  750. state: 'ArrangingPayment',
  751. });
  752. });
  753. it('attempting to add an item returns error result when in ArrangingPayment state', async () => {
  754. const { addItemToOrder } = await shopClient.query<
  755. AddItemToOrder.Mutation,
  756. AddItemToOrder.Variables
  757. >(ADD_ITEM_TO_ORDER, {
  758. productVariantId: 'T_4',
  759. quantity: 1,
  760. });
  761. orderResultGuard.assertErrorResult(addItemToOrder);
  762. expect(addItemToOrder.message).toBe(
  763. `Order contents may only be modified when in the "AddingItems" state`,
  764. );
  765. expect(addItemToOrder.errorCode).toBe(ErrorCode.ORDER_MODIFICATION_ERROR);
  766. });
  767. it('attempting to modify item quantity returns error result when in ArrangingPayment state', async () => {
  768. const { adjustOrderLine } = await shopClient.query<
  769. AdjustItemQuantity.Mutation,
  770. AdjustItemQuantity.Variables
  771. >(ADJUST_ITEM_QUANTITY, {
  772. orderLineId: activeOrder.lines[0].id,
  773. quantity: 12,
  774. });
  775. orderResultGuard.assertErrorResult(adjustOrderLine);
  776. expect(adjustOrderLine.message).toBe(
  777. `Order contents may only be modified when in the "AddingItems" state`,
  778. );
  779. expect(adjustOrderLine.errorCode).toBe(ErrorCode.ORDER_MODIFICATION_ERROR);
  780. });
  781. it('attempting to remove an item returns error result when in ArrangingPayment state', async () => {
  782. const { removeOrderLine } = await shopClient.query<
  783. RemoveItemFromOrder.Mutation,
  784. RemoveItemFromOrder.Variables
  785. >(REMOVE_ITEM_FROM_ORDER, {
  786. orderLineId: activeOrder.lines[0].id,
  787. });
  788. orderResultGuard.assertErrorResult(removeOrderLine);
  789. expect(removeOrderLine.message).toBe(
  790. `Order contents may only be modified when in the "AddingItems" state`,
  791. );
  792. expect(removeOrderLine.errorCode).toBe(ErrorCode.ORDER_MODIFICATION_ERROR);
  793. });
  794. it('attempting to remove all items returns error result when in ArrangingPayment state', async () => {
  795. const { removeAllOrderLines } = await shopClient.query<RemoveAllOrderLines.Mutation>(
  796. REMOVE_ALL_ORDER_LINES,
  797. );
  798. orderResultGuard.assertErrorResult(removeAllOrderLines);
  799. expect(removeAllOrderLines.message).toBe(
  800. `Order contents may only be modified when in the "AddingItems" state`,
  801. );
  802. expect(removeAllOrderLines.errorCode).toBe(ErrorCode.ORDER_MODIFICATION_ERROR);
  803. });
  804. it('attempting to setOrderShippingMethod returns error result when in ArrangingPayment state', async () => {
  805. const shippingMethodsResult = await shopClient.query<GetShippingMethods.Query>(
  806. GET_ELIGIBLE_SHIPPING_METHODS,
  807. );
  808. const shippingMethods = shippingMethodsResult.eligibleShippingMethods;
  809. const { setOrderShippingMethod } = await shopClient.query<
  810. SetShippingMethod.Mutation,
  811. SetShippingMethod.Variables
  812. >(SET_SHIPPING_METHOD, {
  813. id: shippingMethods[0].id,
  814. });
  815. orderResultGuard.assertErrorResult(setOrderShippingMethod);
  816. expect(setOrderShippingMethod.message).toBe(
  817. `Order contents may only be modified when in the "AddingItems" state`,
  818. );
  819. expect(setOrderShippingMethod.errorCode).toBe(ErrorCode.ORDER_MODIFICATION_ERROR);
  820. });
  821. it('adds a declined payment', async () => {
  822. const { addPaymentToOrder } = await shopClient.query<
  823. AddPaymentToOrder.Mutation,
  824. AddPaymentToOrder.Variables
  825. >(ADD_PAYMENT, {
  826. input: {
  827. method: testFailingPaymentMethod.code,
  828. metadata: {
  829. foo: 'bar',
  830. },
  831. },
  832. });
  833. orderResultGuard.assertErrorResult(addPaymentToOrder);
  834. expect(addPaymentToOrder!.message).toBe('The payment was declined');
  835. expect(addPaymentToOrder!.errorCode).toBe(ErrorCode.PAYMENT_DECLINED_ERROR);
  836. expect((addPaymentToOrder as any).paymentErrorMessage).toBe('Insufficient funds');
  837. const { activeOrder: order } = await shopClient.query<GetActiveOrderWithPayments.Query>(
  838. GET_ACTIVE_ORDER_WITH_PAYMENTS,
  839. );
  840. const payment = order!.payments![0];
  841. expect(order!.payments!.length).toBe(1);
  842. expect(payment.method).toBe(testFailingPaymentMethod.code);
  843. expect(payment.state).toBe('Declined');
  844. expect(payment.transactionId).toBe(null);
  845. expect(payment.metadata).toEqual({
  846. foo: 'bar',
  847. });
  848. });
  849. it('adds an error payment and returns error result', async () => {
  850. const { addPaymentToOrder } = await shopClient.query<
  851. AddPaymentToOrder.Mutation,
  852. AddPaymentToOrder.Variables
  853. >(ADD_PAYMENT, {
  854. input: {
  855. method: testErrorPaymentMethod.code,
  856. metadata: {
  857. foo: 'bar',
  858. },
  859. },
  860. });
  861. orderResultGuard.assertErrorResult(addPaymentToOrder);
  862. expect(addPaymentToOrder!.message).toBe('The payment failed');
  863. expect(addPaymentToOrder!.errorCode).toBe(ErrorCode.PAYMENT_FAILED_ERROR);
  864. expect((addPaymentToOrder as any).paymentErrorMessage).toBe('Something went horribly wrong');
  865. const result = await shopClient.query<GetActiveOrderPayments.Query>(
  866. GET_ACTIVE_ORDER_PAYMENTS,
  867. );
  868. const payment = result.activeOrder!.payments![1];
  869. expect(result.activeOrder!.payments!.length).toBe(2);
  870. expect(payment.method).toBe(testErrorPaymentMethod.code);
  871. expect(payment.state).toBe('Error');
  872. expect(payment.errorMessage).toBe('Something went horribly wrong');
  873. });
  874. it('adds a successful payment and transitions Order state', async () => {
  875. const { addPaymentToOrder } = await shopClient.query<
  876. AddPaymentToOrder.Mutation,
  877. AddPaymentToOrder.Variables
  878. >(ADD_PAYMENT, {
  879. input: {
  880. method: testSuccessfulPaymentMethod.code,
  881. metadata: {
  882. baz: 'quux',
  883. },
  884. },
  885. });
  886. orderResultGuard.assertSuccess(addPaymentToOrder);
  887. const payment = addPaymentToOrder!.payments![2];
  888. expect(addPaymentToOrder!.state).toBe('PaymentSettled');
  889. expect(addPaymentToOrder!.active).toBe(false);
  890. expect(addPaymentToOrder!.payments!.length).toBe(3);
  891. expect(payment.method).toBe(testSuccessfulPaymentMethod.code);
  892. expect(payment.state).toBe('Settled');
  893. expect(payment.transactionId).toBe('12345');
  894. expect(payment.metadata).toEqual({
  895. baz: 'quux',
  896. });
  897. });
  898. it('does not create new address when Customer already has address', async () => {
  899. const { customer } = await adminClient.query<GetCustomer.Query, GetCustomer.Variables>(
  900. GET_CUSTOMER,
  901. { id: customers[0].id },
  902. );
  903. expect(customer!.addresses!.length).toBe(1);
  904. });
  905. });
  906. describe('orderByCode', () => {
  907. describe('immediately after Order is placed', () => {
  908. it('works when authenticated', async () => {
  909. const result = await shopClient.query<GetOrderByCode.Query, GetOrderByCode.Variables>(
  910. GET_ORDER_BY_CODE,
  911. {
  912. code: activeOrder.code,
  913. },
  914. );
  915. expect(result.orderByCode!.id).toBe(activeOrder.id);
  916. });
  917. it('works when anonymous', async () => {
  918. await shopClient.asAnonymousUser();
  919. const result = await shopClient.query<GetOrderByCode.Query, GetOrderByCode.Variables>(
  920. GET_ORDER_BY_CODE,
  921. {
  922. code: activeOrder.code,
  923. },
  924. );
  925. expect(result.orderByCode!.id).toBe(activeOrder.id);
  926. });
  927. it(
  928. `throws error for another user's Order`,
  929. assertThrowsWithMessage(async () => {
  930. authenticatedUserEmailAddress = customers[1].emailAddress;
  931. await shopClient.asUserWithCredentials(authenticatedUserEmailAddress, password);
  932. return shopClient.query<GetOrderByCode.Query, GetOrderByCode.Variables>(
  933. GET_ORDER_BY_CODE,
  934. {
  935. code: activeOrder.code,
  936. },
  937. );
  938. }, `You are not currently authorized to perform this action`),
  939. );
  940. });
  941. });
  942. });
  943. describe('order merging', () => {
  944. let customers: GetCustomerList.Items[];
  945. beforeAll(async () => {
  946. const result = await adminClient.query<GetCustomerList.Query>(GET_CUSTOMER_LIST);
  947. customers = result.customers.items;
  948. });
  949. it('merges guest order with no existing order', async () => {
  950. await shopClient.asAnonymousUser();
  951. const { addItemToOrder } = await shopClient.query<
  952. AddItemToOrder.Mutation,
  953. AddItemToOrder.Variables
  954. >(ADD_ITEM_TO_ORDER, {
  955. productVariantId: 'T_1',
  956. quantity: 1,
  957. });
  958. orderResultGuard.assertSuccess(addItemToOrder);
  959. expect(addItemToOrder!.lines.length).toBe(1);
  960. expect(addItemToOrder!.lines[0].productVariant.id).toBe('T_1');
  961. await shopClient.query<AttemptLogin.Mutation, AttemptLogin.Variables>(ATTEMPT_LOGIN, {
  962. username: customers[1].emailAddress,
  963. password: 'test',
  964. });
  965. const { activeOrder } = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  966. expect(activeOrder!.lines.length).toBe(1);
  967. expect(activeOrder!.lines[0].productVariant.id).toBe('T_1');
  968. });
  969. it('merges guest order with existing order', async () => {
  970. await shopClient.asAnonymousUser();
  971. const { addItemToOrder } = await shopClient.query<
  972. AddItemToOrder.Mutation,
  973. AddItemToOrder.Variables
  974. >(ADD_ITEM_TO_ORDER, {
  975. productVariantId: 'T_2',
  976. quantity: 1,
  977. });
  978. orderResultGuard.assertSuccess(addItemToOrder);
  979. expect(addItemToOrder!.lines.length).toBe(1);
  980. expect(addItemToOrder!.lines[0].productVariant.id).toBe('T_2');
  981. await shopClient.query<AttemptLogin.Mutation, AttemptLogin.Variables>(ATTEMPT_LOGIN, {
  982. username: customers[1].emailAddress,
  983. password: 'test',
  984. });
  985. const { activeOrder } = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  986. expect(activeOrder!.lines.length).toBe(2);
  987. expect(activeOrder!.lines[0].productVariant.id).toBe('T_1');
  988. expect(activeOrder!.lines[1].productVariant.id).toBe('T_2');
  989. });
  990. /**
  991. * See https://github.com/vendure-ecommerce/vendure/issues/263
  992. */
  993. it('does not merge when logging in to a different account (issue #263)', async () => {
  994. await shopClient.query<AttemptLogin.Mutation, AttemptLogin.Variables>(ATTEMPT_LOGIN, {
  995. username: customers[2].emailAddress,
  996. password: 'test',
  997. });
  998. const { activeOrder } = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  999. expect(activeOrder).toBeNull();
  1000. });
  1001. it('does not merge when logging back to other account (issue #263)', async () => {
  1002. const { addItemToOrder } = await shopClient.query<
  1003. AddItemToOrder.Mutation,
  1004. AddItemToOrder.Variables
  1005. >(ADD_ITEM_TO_ORDER, {
  1006. productVariantId: 'T_3',
  1007. quantity: 1,
  1008. });
  1009. await shopClient.query<AttemptLogin.Mutation, AttemptLogin.Variables>(ATTEMPT_LOGIN, {
  1010. username: customers[1].emailAddress,
  1011. password: 'test',
  1012. });
  1013. const { activeOrder } = await shopClient.query<GetActiveOrder.Query>(GET_ACTIVE_ORDER);
  1014. expect(activeOrder!.lines.length).toBe(2);
  1015. expect(activeOrder!.lines[0].productVariant.id).toBe('T_1');
  1016. expect(activeOrder!.lines[1].productVariant.id).toBe('T_2');
  1017. });
  1018. });
  1019. describe('security of customer data', () => {
  1020. let customers: GetCustomerList.Items[];
  1021. beforeAll(async () => {
  1022. const result = await adminClient.query<GetCustomerList.Query>(GET_CUSTOMER_LIST);
  1023. customers = result.customers.items;
  1024. });
  1025. it('cannot setCustomOrder to existing non-guest Customer', async () => {
  1026. await shopClient.asAnonymousUser();
  1027. const { addItemToOrder } = await shopClient.query<
  1028. AddItemToOrder.Mutation,
  1029. AddItemToOrder.Variables
  1030. >(ADD_ITEM_TO_ORDER, {
  1031. productVariantId: 'T_1',
  1032. quantity: 1,
  1033. });
  1034. const { setCustomerForOrder } = await shopClient.query<
  1035. SetCustomerForOrder.Mutation,
  1036. SetCustomerForOrder.Variables
  1037. >(SET_CUSTOMER, {
  1038. input: {
  1039. emailAddress: customers[0].emailAddress,
  1040. firstName: 'Evil',
  1041. lastName: 'Hacker',
  1042. },
  1043. });
  1044. orderResultGuard.assertErrorResult(setCustomerForOrder);
  1045. expect(setCustomerForOrder!.message).toBe('The email address is not available.');
  1046. expect(setCustomerForOrder!.errorCode).toBe(ErrorCode.EMAIL_ADDRESS_CONFLICT_ERROR);
  1047. const { customer } = await adminClient.query<GetCustomer.Query, GetCustomer.Variables>(
  1048. GET_CUSTOMER,
  1049. {
  1050. id: customers[0].id,
  1051. },
  1052. );
  1053. expect(customer!.firstName).not.toBe('Evil');
  1054. expect(customer!.lastName).not.toBe('Hacker');
  1055. });
  1056. it('guest cannot access Addresses of guest customer', async () => {
  1057. await shopClient.asAnonymousUser();
  1058. const { addItemToOrder } = await shopClient.query<
  1059. AddItemToOrder.Mutation,
  1060. AddItemToOrder.Variables
  1061. >(ADD_ITEM_TO_ORDER, {
  1062. productVariantId: 'T_1',
  1063. quantity: 1,
  1064. });
  1065. await shopClient.query<SetCustomerForOrder.Mutation, SetCustomerForOrder.Variables>(
  1066. SET_CUSTOMER,
  1067. {
  1068. input: {
  1069. emailAddress: 'test@test.com',
  1070. firstName: 'Evil',
  1071. lastName: 'Hacker',
  1072. },
  1073. },
  1074. );
  1075. const { activeOrder } = await shopClient.query<GetCustomerAddresses.Query>(
  1076. GET_ACTIVE_ORDER_ADDRESSES,
  1077. );
  1078. expect(activeOrder!.customer!.addresses).toEqual([]);
  1079. });
  1080. it('guest cannot access Orders of guest customer', async () => {
  1081. await shopClient.asAnonymousUser();
  1082. const { addItemToOrder } = await shopClient.query<
  1083. AddItemToOrder.Mutation,
  1084. AddItemToOrder.Variables
  1085. >(ADD_ITEM_TO_ORDER, {
  1086. productVariantId: 'T_1',
  1087. quantity: 1,
  1088. });
  1089. await shopClient.query<SetCustomerForOrder.Mutation, SetCustomerForOrder.Variables>(
  1090. SET_CUSTOMER,
  1091. {
  1092. input: {
  1093. emailAddress: 'test@test.com',
  1094. firstName: 'Evil',
  1095. lastName: 'Hacker',
  1096. },
  1097. },
  1098. );
  1099. const { activeOrder } = await shopClient.query<GetCustomerOrders.Query>(GET_ACTIVE_ORDER_ORDERS);
  1100. expect(activeOrder!.customer!.orders.items).toEqual([]);
  1101. });
  1102. });
  1103. describe('order custom fields', () => {
  1104. it('custom fields added to type', async () => {
  1105. await shopClient.asAnonymousUser();
  1106. await shopClient.query<AddItemToOrder.Mutation, AddItemToOrder.Variables>(ADD_ITEM_TO_ORDER, {
  1107. productVariantId: 'T_1',
  1108. quantity: 1,
  1109. });
  1110. const { activeOrder } = await shopClient.query(GET_ORDER_CUSTOM_FIELDS);
  1111. expect(activeOrder?.customFields).toEqual({
  1112. giftWrap: false,
  1113. });
  1114. });
  1115. it('setting order custom fields', async () => {
  1116. const { setOrderCustomFields } = await shopClient.query(SET_ORDER_CUSTOM_FIELDS, {
  1117. input: {
  1118. customFields: { giftWrap: true },
  1119. },
  1120. });
  1121. expect(setOrderCustomFields?.customFields).toEqual({
  1122. giftWrap: true,
  1123. });
  1124. const { activeOrder } = await shopClient.query(GET_ORDER_CUSTOM_FIELDS);
  1125. expect(activeOrder?.customFields).toEqual({
  1126. giftWrap: true,
  1127. });
  1128. });
  1129. });
  1130. describe('remove all order lines', () => {
  1131. beforeAll(async () => {
  1132. await shopClient.asAnonymousUser();
  1133. await shopClient.query<AddItemToOrder.Mutation, AddItemToOrder.Variables>(ADD_ITEM_TO_ORDER, {
  1134. productVariantId: 'T_1',
  1135. quantity: 1,
  1136. });
  1137. await shopClient.query<AddItemToOrder.Mutation, AddItemToOrder.Variables>(ADD_ITEM_TO_ORDER, {
  1138. productVariantId: 'T_2',
  1139. quantity: 3,
  1140. });
  1141. });
  1142. it('should remove all order lines', async () => {
  1143. const { removeAllOrderLines } = await shopClient.query<
  1144. RemoveAllOrderLines.Mutation,
  1145. RemoveAllOrderLines.Variables
  1146. >(REMOVE_ALL_ORDER_LINES);
  1147. orderResultGuard.assertSuccess(removeAllOrderLines);
  1148. expect(removeAllOrderLines?.total).toBe(0);
  1149. expect(removeAllOrderLines?.lines.length).toBe(0);
  1150. });
  1151. });
  1152. });
  1153. const GET_ORDER_CUSTOM_FIELDS = gql`
  1154. query GetOrderCustomFields {
  1155. activeOrder {
  1156. id
  1157. customFields {
  1158. giftWrap
  1159. }
  1160. }
  1161. }
  1162. `;
  1163. const SET_ORDER_CUSTOM_FIELDS = gql`
  1164. mutation SetOrderCustomFields($input: UpdateOrderInput!) {
  1165. setOrderCustomFields(input: $input) {
  1166. id
  1167. customFields {
  1168. giftWrap
  1169. }
  1170. }
  1171. }
  1172. `;