draft-order.e2e-spec.ts 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709
  1. /* eslint-disable @typescript-eslint/no-non-null-assertion */
  2. import { LanguageCode } from '@vendure/common/lib/generated-types';
  3. import {
  4. DefaultOrderPlacedStrategy,
  5. mergeConfig,
  6. Order,
  7. orderPercentageDiscount,
  8. OrderState,
  9. RequestContext,
  10. } from '@vendure/core';
  11. import { createErrorResultGuard, createTestEnvironment, ErrorResultGuard } from '@vendure/testing';
  12. import path from 'path';
  13. import { afterAll, beforeAll, describe, expect, it, vi } from 'vitest';
  14. import { initialData } from '../../../e2e-common/e2e-initial-data';
  15. import { TEST_SETUP_TIMEOUT_MS, testConfig } from '../../../e2e-common/test-config';
  16. import { singleStageRefundablePaymentMethod } from './fixtures/test-payment-methods';
  17. import {
  18. canceledOrderFragment,
  19. orderFragment,
  20. orderWithLinesFragment,
  21. paymentFragment,
  22. } from './graphql/fragments-admin';
  23. import { FragmentOf, graphql, ResultOf } from './graphql/graphql-admin';
  24. import {
  25. addManualPaymentDocument,
  26. adminTransitionToStateDocument,
  27. createPromotionDocument,
  28. getCustomerListDocument,
  29. } from './graphql/shared-definitions';
  30. import { getActiveCustomerOrdersDocument } from './graphql/shop-definitions';
  31. class TestOrderPlacedStrategy extends DefaultOrderPlacedStrategy {
  32. static spy = vi.fn();
  33. shouldSetAsPlaced(
  34. ctx: RequestContext,
  35. fromState: OrderState,
  36. toState: OrderState,
  37. order: Order,
  38. ): boolean {
  39. TestOrderPlacedStrategy.spy(order);
  40. return super.shouldSetAsPlaced(ctx, fromState, toState, order);
  41. }
  42. }
  43. describe('Draft Orders resolver', () => {
  44. const { server, adminClient, shopClient } = createTestEnvironment(
  45. mergeConfig(testConfig(), {
  46. paymentOptions: {
  47. paymentMethodHandlers: [singleStageRefundablePaymentMethod],
  48. },
  49. orderOptions: {
  50. orderPlacedStrategy: new TestOrderPlacedStrategy(),
  51. },
  52. dbConnectionOptions: {
  53. logging: true,
  54. },
  55. }),
  56. );
  57. let customers: ResultOf<typeof getCustomerListDocument>['customers']['items'];
  58. let draftOrder: FragmentOf<typeof orderWithLinesFragment>;
  59. const freeOrderCouponCode = 'FREE';
  60. const orderGuard: ErrorResultGuard<
  61. | FragmentOf<typeof orderWithLinesFragment>
  62. | FragmentOf<typeof canceledOrderFragment>
  63. | FragmentOf<typeof orderFragment>
  64. > = createErrorResultGuard(input => ('lines' in input && !!input.lines) || !!input.state);
  65. beforeAll(async () => {
  66. await server.init({
  67. initialData: {
  68. ...initialData,
  69. paymentMethods: [
  70. {
  71. name: singleStageRefundablePaymentMethod.code,
  72. handler: { code: singleStageRefundablePaymentMethod.code, arguments: [] },
  73. },
  74. ],
  75. },
  76. productsCsvPath: path.join(__dirname, 'fixtures/e2e-products-full.csv'),
  77. customerCount: 3,
  78. });
  79. await adminClient.asSuperAdmin();
  80. // Create a couple of orders to be queried
  81. const result = await adminClient.query(getCustomerListDocument, {
  82. options: {
  83. take: 3,
  84. },
  85. });
  86. customers = result.customers.items;
  87. // Create a coupon code promotion
  88. await adminClient.query(createPromotionDocument, {
  89. input: {
  90. enabled: true,
  91. conditions: [],
  92. couponCode: freeOrderCouponCode,
  93. actions: [
  94. {
  95. code: orderPercentageDiscount.code,
  96. arguments: [{ name: 'discount', value: '100' }],
  97. },
  98. ],
  99. translations: [{ languageCode: LanguageCode.en, name: 'Free Order' }],
  100. },
  101. });
  102. }, TEST_SETUP_TIMEOUT_MS);
  103. afterAll(async () => {
  104. await server.destroy();
  105. });
  106. it('create draft order', async () => {
  107. const { createDraftOrder } = await adminClient.query(createDraftOrderDocument);
  108. expect(createDraftOrder.state).toBe('Draft');
  109. expect(createDraftOrder.active).toBe(false);
  110. draftOrder = createDraftOrder;
  111. });
  112. it('addItemToDraftOrder', async () => {
  113. const { addItemToDraftOrder } = await adminClient.query(addItemToDraftOrderDocument, {
  114. orderId: draftOrder.id,
  115. input: {
  116. productVariantId: 'T_5',
  117. quantity: 2,
  118. },
  119. });
  120. orderGuard.assertSuccess(addItemToDraftOrder);
  121. expect(addItemToDraftOrder.lines.length).toBe(1);
  122. draftOrder = addItemToDraftOrder;
  123. });
  124. it('adjustDraftOrderLine up', async () => {
  125. const firstLine = draftOrder.lines[0];
  126. if (!firstLine) throw new Error('Expected first line to exist');
  127. const { adjustDraftOrderLine } = await adminClient.query(adjustDraftOrderLineDocument, {
  128. orderId: draftOrder.id,
  129. input: {
  130. orderLineId: firstLine.id,
  131. quantity: 5,
  132. },
  133. });
  134. orderGuard.assertSuccess(adjustDraftOrderLine);
  135. expect(adjustDraftOrderLine.lines[0].quantity).toBe(5);
  136. });
  137. it('adjustDraftOrderLine down', async () => {
  138. const firstLine = draftOrder.lines[0];
  139. if (!firstLine) throw new Error('Expected first line to exist');
  140. const { adjustDraftOrderLine } = await adminClient.query(adjustDraftOrderLineDocument, {
  141. orderId: draftOrder.id,
  142. input: {
  143. orderLineId: firstLine.id,
  144. quantity: 2,
  145. },
  146. });
  147. orderGuard.assertSuccess(adjustDraftOrderLine);
  148. expect(adjustDraftOrderLine.lines[0].quantity).toBe(2);
  149. });
  150. it('removeDraftOrderLine', async () => {
  151. const firstLine = draftOrder.lines[0];
  152. if (!firstLine) throw new Error('Expected first line to exist');
  153. const { removeDraftOrderLine } = await adminClient.query(removeDraftOrderLineDocument, {
  154. orderId: draftOrder.id,
  155. orderLineId: firstLine.id,
  156. });
  157. orderGuard.assertSuccess(removeDraftOrderLine);
  158. expect(removeDraftOrderLine.lines.length).toBe(0);
  159. });
  160. it('setCustomerForDraftOrder', async () => {
  161. const { setCustomerForDraftOrder } = await adminClient.query(setCustomerForDraftOrderDocument, {
  162. orderId: draftOrder.id,
  163. customerId: customers[0].id,
  164. });
  165. orderGuard.assertSuccess(setCustomerForDraftOrder);
  166. expect(setCustomerForDraftOrder.customer?.id).toBe(customers[0].id);
  167. });
  168. it('custom does not see draft orders in history', async () => {
  169. await shopClient.asUserWithCredentials(customers[0].emailAddress, 'test');
  170. const { activeCustomer } = await shopClient.query(getActiveCustomerOrdersDocument);
  171. expect(activeCustomer?.orders.totalItems).toBe(0);
  172. expect(activeCustomer?.orders.items.length).toBe(0);
  173. });
  174. it('setDraftOrderShippingAddress', async () => {
  175. const { setDraftOrderShippingAddress } = await adminClient.query(
  176. setShippingAddressForDraftOrderDocument,
  177. {
  178. orderId: draftOrder.id,
  179. input: {
  180. streetLine1: 'Shipping Street',
  181. city: 'Wigan',
  182. province: 'Greater Manchester',
  183. postalCode: 'WN1 2DD',
  184. countryCode: 'GB',
  185. },
  186. },
  187. );
  188. expect(setDraftOrderShippingAddress.shippingAddress).toEqual({
  189. company: null,
  190. fullName: null,
  191. phoneNumber: null,
  192. streetLine2: null,
  193. province: 'Greater Manchester',
  194. city: 'Wigan',
  195. country: 'United Kingdom',
  196. postalCode: 'WN1 2DD',
  197. streetLine1: 'Shipping Street',
  198. });
  199. });
  200. it('setDraftOrderBillingAddress', async () => {
  201. const { setDraftOrderBillingAddress } = await adminClient.query(
  202. setBillingAddressForDraftOrderDocument,
  203. {
  204. orderId: draftOrder.id,
  205. input: {
  206. streetLine1: 'Billing Street',
  207. city: 'Skelmerdale',
  208. province: 'Lancashire',
  209. postalCode: 'WN8 3QW',
  210. countryCode: 'GB',
  211. },
  212. },
  213. );
  214. expect(setDraftOrderBillingAddress.billingAddress).toEqual({
  215. company: null,
  216. fullName: null,
  217. phoneNumber: null,
  218. streetLine2: null,
  219. province: 'Lancashire',
  220. city: 'Skelmerdale',
  221. country: 'United Kingdom',
  222. postalCode: 'WN8 3QW',
  223. streetLine1: 'Billing Street',
  224. });
  225. });
  226. it('unsetDraftOrderShippingAddress', async () => {
  227. const { unsetDraftOrderShippingAddress } = await adminClient.query(
  228. unsetShippingAddressForDraftOrderDocument,
  229. {
  230. orderId: draftOrder.id,
  231. },
  232. );
  233. expect(unsetDraftOrderShippingAddress.shippingAddress).toEqual({
  234. company: null,
  235. fullName: null,
  236. phoneNumber: null,
  237. streetLine2: null,
  238. province: null,
  239. city: null,
  240. country: null,
  241. postalCode: null,
  242. streetLine1: null,
  243. });
  244. });
  245. it('unsetDraftOrderBillingAddress', async () => {
  246. const { unsetDraftOrderBillingAddress } = await adminClient.query(
  247. unsetBillingAddressForDraftOrderDocument,
  248. {
  249. orderId: draftOrder.id,
  250. },
  251. );
  252. expect(unsetDraftOrderBillingAddress.billingAddress).toEqual({
  253. company: null,
  254. fullName: null,
  255. phoneNumber: null,
  256. streetLine2: null,
  257. province: null,
  258. city: null,
  259. country: null,
  260. postalCode: null,
  261. streetLine1: null,
  262. });
  263. });
  264. it('applyCouponCodeToDraftOrder', async () => {
  265. const { addItemToDraftOrder } = await adminClient.query(addItemToDraftOrderDocument, {
  266. orderId: draftOrder.id,
  267. input: {
  268. productVariantId: 'T_1',
  269. quantity: 1,
  270. },
  271. });
  272. orderGuard.assertSuccess(addItemToDraftOrder);
  273. expect(addItemToDraftOrder.totalWithTax).toBe(155880);
  274. const { applyCouponCodeToDraftOrder } = await adminClient.query(applyCouponCodeToDraftOrderDocument, {
  275. orderId: draftOrder.id,
  276. couponCode: freeOrderCouponCode,
  277. });
  278. orderGuard.assertSuccess(applyCouponCodeToDraftOrder);
  279. expect(applyCouponCodeToDraftOrder.couponCodes).toEqual([freeOrderCouponCode]);
  280. expect(applyCouponCodeToDraftOrder.totalWithTax).toBe(0);
  281. });
  282. it('removeCouponCodeFromDraftOrder', async () => {
  283. const { removeCouponCodeFromDraftOrder } = await adminClient.query(
  284. removeCouponCodeFromDraftOrderDocument,
  285. {
  286. orderId: draftOrder.id,
  287. couponCode: freeOrderCouponCode,
  288. },
  289. );
  290. if (!removeCouponCodeFromDraftOrder) throw new Error('Expected order to be returned');
  291. expect(removeCouponCodeFromDraftOrder.couponCodes).toEqual([]);
  292. expect(removeCouponCodeFromDraftOrder.totalWithTax).toBe(155880);
  293. });
  294. it('eligibleShippingMethodsForDraftOrder', async () => {
  295. const { eligibleShippingMethodsForDraftOrder } = await adminClient.query(
  296. draftOrderEligibleShippingMethodsDocument,
  297. {
  298. orderId: draftOrder.id,
  299. },
  300. );
  301. expect(eligibleShippingMethodsForDraftOrder).toEqual([
  302. {
  303. code: 'standard-shipping',
  304. description: '',
  305. id: 'T_1',
  306. metadata: null,
  307. name: 'Standard Shipping',
  308. price: 500,
  309. priceWithTax: 500,
  310. },
  311. {
  312. code: 'express-shipping',
  313. description: '',
  314. id: 'T_2',
  315. metadata: null,
  316. name: 'Express Shipping',
  317. price: 1000,
  318. priceWithTax: 1000,
  319. },
  320. {
  321. code: 'express-shipping-taxed',
  322. description: '',
  323. id: 'T_3',
  324. metadata: null,
  325. name: 'Express Shipping (Taxed)',
  326. price: 1000,
  327. priceWithTax: 1200,
  328. },
  329. ]);
  330. });
  331. it('setDraftOrderShippingMethod', async () => {
  332. const { setDraftOrderShippingMethod } = await adminClient.query(setDraftOrderShippingMethodDocument, {
  333. orderId: draftOrder.id,
  334. shippingMethodId: 'T_2',
  335. });
  336. orderGuard.assertSuccess(setDraftOrderShippingMethod);
  337. expect(setDraftOrderShippingMethod.shippingWithTax).toBe(1000);
  338. expect(setDraftOrderShippingMethod.shippingLines.length).toBe(1);
  339. expect(setDraftOrderShippingMethod.shippingLines[0].shippingMethod.id).toBe('T_2');
  340. });
  341. // https://github.com/vendure-ecommerce/vendure/issues/2105
  342. it('sets order as placed when payment is settled', async () => {
  343. TestOrderPlacedStrategy.spy.mockClear();
  344. expect(TestOrderPlacedStrategy.spy.mock.calls.length).toBe(0);
  345. const { transitionOrderToState } = await adminClient.query(adminTransitionToStateDocument, {
  346. id: draftOrder.id,
  347. state: 'ArrangingPayment',
  348. });
  349. if (!transitionOrderToState) throw new Error('Expected transitionOrderToState result');
  350. orderGuard.assertSuccess(transitionOrderToState);
  351. expect(transitionOrderToState.state).toBe('ArrangingPayment');
  352. const { addManualPaymentToOrder } = await adminClient.query(addManualPaymentDocument, {
  353. input: {
  354. orderId: draftOrder.id,
  355. metadata: {},
  356. method: singleStageRefundablePaymentMethod.code,
  357. transactionId: '12345',
  358. },
  359. });
  360. orderGuard.assertSuccess(addManualPaymentToOrder);
  361. expect(addManualPaymentToOrder.state).toBe('PaymentSettled');
  362. const { order } = await adminClient.query(getOrderPlacedAtDocument, {
  363. id: draftOrder.id,
  364. });
  365. expect(order?.orderPlacedAt).not.toBeNull();
  366. expect(TestOrderPlacedStrategy.spy.mock.calls.length).toBe(1);
  367. expect(TestOrderPlacedStrategy.spy.mock.calls[0][0].code).toBe(draftOrder.code);
  368. });
  369. });
  370. // Local fragment without countryCode to match original test expectations
  371. const draftOrderShippingAddressFragment = graphql(`
  372. fragment DraftOrderShippingAddress on OrderAddress {
  373. fullName
  374. company
  375. streetLine1
  376. streetLine2
  377. city
  378. province
  379. postalCode
  380. country
  381. phoneNumber
  382. }
  383. `);
  384. // Local orderWithLines fragment using the address fragment without countryCode
  385. const draftOrderWithLinesFragment = graphql(
  386. `
  387. fragment DraftOrderWithLines on Order {
  388. id
  389. createdAt
  390. updatedAt
  391. code
  392. state
  393. active
  394. customer {
  395. id
  396. firstName
  397. lastName
  398. }
  399. lines {
  400. id
  401. featuredAsset {
  402. preview
  403. }
  404. productVariant {
  405. id
  406. name
  407. sku
  408. }
  409. taxLines {
  410. description
  411. taxRate
  412. }
  413. unitPrice
  414. unitPriceWithTax
  415. quantity
  416. taxRate
  417. linePriceWithTax
  418. }
  419. surcharges {
  420. id
  421. description
  422. sku
  423. price
  424. priceWithTax
  425. }
  426. subTotal
  427. subTotalWithTax
  428. total
  429. totalWithTax
  430. totalQuantity
  431. currencyCode
  432. shipping
  433. shippingWithTax
  434. shippingLines {
  435. priceWithTax
  436. shippingMethod {
  437. id
  438. code
  439. name
  440. description
  441. }
  442. }
  443. shippingAddress {
  444. ...DraftOrderShippingAddress
  445. }
  446. payments {
  447. ...Payment
  448. }
  449. fulfillments {
  450. id
  451. state
  452. method
  453. trackingCode
  454. lines {
  455. orderLineId
  456. quantity
  457. }
  458. }
  459. }
  460. `,
  461. [draftOrderShippingAddressFragment, paymentFragment],
  462. );
  463. const createDraftOrderDocument = graphql(
  464. `
  465. mutation CreateDraftOrder {
  466. createDraftOrder {
  467. ...OrderWithLines
  468. }
  469. }
  470. `,
  471. [orderWithLinesFragment],
  472. );
  473. const addItemToDraftOrderDocument = graphql(
  474. `
  475. mutation AddItemToDraftOrder($orderId: ID!, $input: AddItemToDraftOrderInput!) {
  476. addItemToDraftOrder(orderId: $orderId, input: $input) {
  477. ...OrderWithLines
  478. ... on ErrorResult {
  479. errorCode
  480. message
  481. }
  482. }
  483. }
  484. `,
  485. [orderWithLinesFragment],
  486. );
  487. const adjustDraftOrderLineDocument = graphql(
  488. `
  489. mutation AdjustDraftOrderLine($orderId: ID!, $input: AdjustDraftOrderLineInput!) {
  490. adjustDraftOrderLine(orderId: $orderId, input: $input) {
  491. ...OrderWithLines
  492. ... on ErrorResult {
  493. errorCode
  494. message
  495. }
  496. }
  497. }
  498. `,
  499. [orderWithLinesFragment],
  500. );
  501. const removeDraftOrderLineDocument = graphql(
  502. `
  503. mutation RemoveDraftOrderLine($orderId: ID!, $orderLineId: ID!) {
  504. removeDraftOrderLine(orderId: $orderId, orderLineId: $orderLineId) {
  505. ...OrderWithLines
  506. ... on ErrorResult {
  507. errorCode
  508. message
  509. }
  510. }
  511. }
  512. `,
  513. [orderWithLinesFragment],
  514. );
  515. const setCustomerForDraftOrderDocument = graphql(
  516. `
  517. mutation SetCustomerForDraftOrder($orderId: ID!, $customerId: ID, $input: CreateCustomerInput) {
  518. setCustomerForDraftOrder(orderId: $orderId, customerId: $customerId, input: $input) {
  519. ...OrderWithLines
  520. ... on ErrorResult {
  521. errorCode
  522. message
  523. }
  524. }
  525. }
  526. `,
  527. [orderWithLinesFragment],
  528. );
  529. const setShippingAddressForDraftOrderDocument = graphql(
  530. `
  531. mutation SetDraftOrderShippingAddress($orderId: ID!, $input: CreateAddressInput!) {
  532. setDraftOrderShippingAddress(orderId: $orderId, input: $input) {
  533. ...DraftOrderWithLines
  534. }
  535. }
  536. `,
  537. [draftOrderWithLinesFragment],
  538. );
  539. const setBillingAddressForDraftOrderDocument = graphql(
  540. `
  541. mutation SetDraftOrderBillingAddress($orderId: ID!, $input: CreateAddressInput!) {
  542. setDraftOrderBillingAddress(orderId: $orderId, input: $input) {
  543. ...DraftOrderWithLines
  544. billingAddress {
  545. ...DraftOrderShippingAddress
  546. }
  547. }
  548. }
  549. `,
  550. [draftOrderWithLinesFragment, draftOrderShippingAddressFragment],
  551. );
  552. const unsetShippingAddressForDraftOrderDocument = graphql(
  553. `
  554. mutation UnsetDraftOrderShippingAddress($orderId: ID!) {
  555. unsetDraftOrderShippingAddress(orderId: $orderId) {
  556. ...DraftOrderWithLines
  557. }
  558. }
  559. `,
  560. [draftOrderWithLinesFragment],
  561. );
  562. const unsetBillingAddressForDraftOrderDocument = graphql(
  563. `
  564. mutation UnsetDraftOrderBillingAddress($orderId: ID!) {
  565. unsetDraftOrderBillingAddress(orderId: $orderId) {
  566. ...DraftOrderWithLines
  567. billingAddress {
  568. ...DraftOrderShippingAddress
  569. }
  570. }
  571. }
  572. `,
  573. [draftOrderWithLinesFragment, draftOrderShippingAddressFragment],
  574. );
  575. const applyCouponCodeToDraftOrderDocument = graphql(
  576. `
  577. mutation ApplyCouponCodeToDraftOrder($orderId: ID!, $couponCode: String!) {
  578. applyCouponCodeToDraftOrder(orderId: $orderId, couponCode: $couponCode) {
  579. ...OrderWithLines
  580. ... on Order {
  581. couponCodes
  582. }
  583. ... on ErrorResult {
  584. errorCode
  585. message
  586. }
  587. }
  588. }
  589. `,
  590. [orderWithLinesFragment],
  591. );
  592. const removeCouponCodeFromDraftOrderDocument = graphql(
  593. `
  594. mutation RemoveCouponCodeFromDraftOrder($orderId: ID!, $couponCode: String!) {
  595. removeCouponCodeFromDraftOrder(orderId: $orderId, couponCode: $couponCode) {
  596. ...OrderWithLines
  597. ... on Order {
  598. couponCodes
  599. }
  600. }
  601. }
  602. `,
  603. [orderWithLinesFragment],
  604. );
  605. const draftOrderEligibleShippingMethodsDocument = graphql(`
  606. query DraftOrderEligibleShippingMethods($orderId: ID!) {
  607. eligibleShippingMethodsForDraftOrder(orderId: $orderId) {
  608. id
  609. name
  610. code
  611. description
  612. price
  613. priceWithTax
  614. metadata
  615. }
  616. }
  617. `);
  618. const setDraftOrderShippingMethodDocument = graphql(
  619. `
  620. mutation SetDraftOrderShippingMethod($orderId: ID!, $shippingMethodId: ID!) {
  621. setDraftOrderShippingMethod(orderId: $orderId, shippingMethodId: $shippingMethodId) {
  622. ...OrderWithLines
  623. ... on ErrorResult {
  624. errorCode
  625. message
  626. }
  627. }
  628. }
  629. `,
  630. [orderWithLinesFragment],
  631. );
  632. const getOrderPlacedAtDocument = graphql(`
  633. query GetOrderPlacedAt($id: ID!) {
  634. order(id: $id) {
  635. id
  636. createdAt
  637. updatedAt
  638. state
  639. orderPlacedAt
  640. }
  641. }
  642. `);