Интеграция Saleor GraphQL API с фронтендом

Наша компания занимается разработкой, поддержкой и обслуживанием сайтов любой сложности. От простых одностраничных сайтов до масштабных кластерных систем построенных на микро сервисах. Опыт разработчиков подтвержден сертификатами от вендоров.

Разработка и обслуживание любых видов сайтов:

Информационные сайты или веб-приложения
Сайты визитки, landing page, корпоративные сайты, онлайн каталоги, квиз, промо-сайты, блоги, новостные ресурсы, информационные порталы, форумы, агрегаторы
Сайты или веб-приложения электронной коммерции
Интернет-магазины, B2B-порталы, маркетплейсы, онлайн-обменники, кэшбэк-сайты, биржи, дропшиппинг-платформы, парсеры товаров
Веб-приложения для управления бизнес-процессами
CRM-системы, ERP-системы, корпоративные порталы, системы управления производством, парсеры информации
Сайты или веб-приложения электронных услуг
Доски объявлений, онлайн-школы, онлайн-кинотеатры, конструкторы сайтов, порталы предоставления электронных услуг, видеохостинги, тематические порталы

Это лишь некоторые из технических типов сайтов, с которыми мы работаем, и каждый из них может иметь свои специфические особенности и функциональность, а также быть адаптированным под конкретные потребности и цели клиента

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Интеграция Saleor GraphQL API с фронтендом
Средняя
~5 рабочих дней
Часто задаваемые вопросы

Наши компетенции:

Этапы разработки

Последние работы

  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1262
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1171
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    874
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1094
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    831
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    851

Интеграция Saleor GraphQL API с фронтендом

Saleor предоставляет единый GraphQL endpoint для всех операций — каталог, корзина, checkout, платежи, аккаунт. Фронтенд работает напрямую с этим API без промежуточного REST слоя. Интеграция строится на Apollo Client или urql, с генерацией типов через @graphql-codegen.

Настройка Apollo Client

// lib/apolloClient.ts
import {
  ApolloClient,
  InMemoryCache,
  createHttpLink,
  from,
} from "@apollo/client";
import { setContext } from "@apollo/client/link/context";
import { onError } from "@apollo/client/link/error";

const httpLink = createHttpLink({
  uri: process.env.NEXT_PUBLIC_SALEOR_API_URL,
});

const authLink = setContext((_, { headers }) => {
  const token = localStorage.getItem("saleor_token");
  return {
    headers: {
      ...headers,
      authorization: token ? `Bearer ${token}` : "",
    },
  };
});

const errorLink = onError(({ graphQLErrors, networkError }) => {
  if (graphQLErrors) {
    graphQLErrors.forEach(({ message, extensions }) => {
      if (extensions?.code === "AUTHENTICATION_FAILED") {
        localStorage.removeItem("saleor_token");
        window.location.href = "/login";
      }
    });
  }
});

export const client = new ApolloClient({
  link: from([errorLink, authLink, httpLink]),
  cache: new InMemoryCache({
    typePolicies: {
      Product: { keyFields: ["id"] },
      ProductVariant: { keyFields: ["id"] },
      Checkout: { keyFields: ["id"] },
    },
  }),
});

Генерация типов через codegen

# codegen.yml
overwrite: true
schema: "https://api.your-store.com/graphql/"
documents: "src/**/*.graphql"
generates:
  src/generated/graphql.ts:
    plugins:
      - typescript
      - typescript-operations
      - typescript-react-apollo
    config:
      withHooks: true
      withComponent: false
      scalars:
        JSON: "Record<string, unknown>"
        Date: "string"
        Decimal: "string"
        UUID: "string"
        PositiveDecimal: "number"
npx graphql-codegen --config codegen.yml

Результат — полностью типизированные хуки useProductListQuery, useCheckoutCreateMutation и т.д.

Каталог: список товаров с пагинацией

# queries/products.graphql
query ProductList(
  $first: Int
  $after: String
  $filter: ProductFilterInput
  $channel: String!
) {
  products(first: $first, after: $after, filter: $filter, channel: $channel) {
    edges {
      node {
        id
        name
        slug
        thumbnail { url alt }
        pricing {
          priceRange {
            start { gross { amount currency } }
          }
        }
      }
    }
    pageInfo {
      hasNextPage
      endCursor
    }
  }
}
const { data, fetchMore } = useProductListQuery({
  variables: { first: 24, channel: "default-channel" },
});

const loadMore = () => {
  fetchMore({
    variables: { after: data?.products?.pageInfo.endCursor },
    updateQuery: (prev, { fetchMoreResult }) => {
      if (!fetchMoreResult) return prev;
      return {
        products: {
          ...fetchMoreResult.products,
          edges: [
            ...prev.products!.edges,
            ...fetchMoreResult.products!.edges,
          ],
        },
      };
    },
  });
};

Checkout flow

Saleor разделяет checkout на явные мутации. Полный flow:

// 1. Создать checkout
const [createCheckout] = useCheckoutCreateMutation();
const { data } = await createCheckout({
  variables: {
    input: {
      channel: "default-channel",
      lines: [{ variantId, quantity: 1 }],
      email: "[email protected]",
    },
  },
});
const checkoutId = data?.checkoutCreate?.checkout?.id;

// 2. Добавить адрес доставки
const [updateShippingAddress] = useCheckoutShippingAddressUpdateMutation();
await updateShippingAddress({
  variables: {
    id: checkoutId,
    shippingAddress: {
      firstName: "Ivan",
      lastName: "Petrov",
      streetAddress1: "ul. Lenina 1",
      city: "Moscow",
      country: CountryCode.Ru,
      postalCode: "101000",
    },
  },
});

// 3. Выбрать метод доставки
const [updateDelivery] = useCheckoutDeliveryMethodUpdateMutation();
await updateDelivery({
  variables: { id: checkoutId, deliveryMethodId: shippingMethodId },
});

// 4. Создать платёж
const [createPayment] = useCheckoutPaymentCreateMutation();
await createPayment({
  variables: {
    id: checkoutId,
    input: {
      gateway: "mirumee.payments.stripe",
      token: stripeToken,
      amount: checkoutTotal,
    },
  },
});

// 5. Завершить заказ
const [completeCheckout] = useCheckoutCompleteMutation();
const order = await completeCheckout({ variables: { id: checkoutId } });

Аутентификация пользователей

// Логин
const [tokenCreate] = useTokenCreateMutation();
const { data } = await tokenCreate({
  variables: { email, password },
});
const { token, refreshToken, errors } = data!.tokenCreate!;

if (!errors?.length) {
  localStorage.setItem("saleor_token", token!);
  localStorage.setItem("saleor_refresh_token", refreshToken!);
}

// Обновление токена
const [tokenRefresh] = useTokenRefreshMutation();
const refreshed = await tokenRefresh({
  variables: { token: localStorage.getItem("saleor_refresh_token")! },
});

Обработка ошибок Saleor

Saleor возвращает ошибки не через стандартный GraphQL errors, а через поле errors в теле ответа мутации. Паттерн обработки:

function handleSaleorErrors<T extends { errors: SaleorError[] }>(
  result: T | null | undefined,
  onSuccess: (data: T) => void
) {
  if (!result) return;
  if (result.errors.length > 0) {
    result.errors.forEach((err) => {
      console.error(`${err.field}: ${err.message} (${err.code})`);
    });
    return;
  }
  onSuccess(result);
}

Производительность

  • Saleor поддерживает persisted queries — передавать hash вместо текста запроса
  • Используйте fragments для переиспользования полей между запросами
  • InMemoryCache с правильными keyFields исключает дублирование данных
  • Для SSR (Next.js) — @apollo/experimental-nextjs-app-support или getStaticProps с client.query()

Сроки интеграции

Этап Срок
Настройка Apollo Client + codegen 1 день
Каталог (список, фильтры, страница товара) 2–3 дня
Корзина + checkout (без оплаты) 2–3 дня
Платёжный gateway (Stripe/Adyen) 2–3 дня
Аккаунт пользователя, история заказов 1–2 дня