Реализация SDK/библиотеки для интеграций с SaaS-приложением

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

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

Предлагаемые услуги
Показано 1 из 1 услугВсе 2065 услуг
Реализация SDK/библиотеки для интеграций с SaaS-приложением
Сложная
~2-4 недели
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • 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

Реализация SDK/библиотеки для интеграций с SaaS-приложением

SDK (Software Development Kit) — готовая библиотека для клиентов API. Снижает барьер интеграции: вместо работы с HTTP напрямую разработчик использует типизированные методы.

Структура SDK-проекта

my-api-sdk/
  src/
    client.ts          # Основной HTTP-клиент
    resources/
      users.ts         # Ресурс: пользователи
      projects.ts      # Ресурс: проекты
      webhooks.ts      # Верификация webhook
    types/
      index.ts         # Все публичные типы
      responses.ts     # Типы ответов API
    errors.ts          # Кастомные ошибки
    retry.ts           # Retry логика
    pagination.ts      # Pager для списков
  tests/
    client.test.ts
  dist/                # Скомпилированный JS + типы
  package.json
  tsconfig.json
  README.md

HTTP-клиент

// src/client.ts
export interface MyApiConfig {
  apiKey: string;
  baseUrl?: string;
  timeout?: number;
  maxRetries?: number;
}

export class MyApiError extends Error {
  constructor(
    message: string,
    public readonly statusCode: number,
    public readonly code: string,
    public readonly requestId: string
  ) {
    super(message);
    this.name = 'MyApiError';
  }
}

export class MyApiClient {
  private readonly baseUrl: string;
  private readonly apiKey: string;
  private readonly timeout: number;
  private readonly maxRetries: number;

  // Ресурсы
  public readonly users: UsersResource;
  public readonly projects: ProjectsResource;
  public readonly webhooks: WebhooksResource;

  constructor(config: MyApiConfig) {
    this.baseUrl = config.baseUrl ?? 'https://api.myproduct.com/v1';
    this.apiKey = config.apiKey;
    this.timeout = config.timeout ?? 30_000;
    this.maxRetries = config.maxRetries ?? 3;

    this.users = new UsersResource(this);
    this.projects = new ProjectsResource(this);
    this.webhooks = new WebhooksResource(this);
  }

  async request<T>(
    method: string,
    path: string,
    options: RequestOptions = {}
  ): Promise<T> {
    const url = `${this.baseUrl}${path}`;
    const headers: Record<string, string> = {
      'Authorization': `Bearer ${this.apiKey}`,
      'Content-Type': 'application/json',
      'User-Agent': `myapi-sdk-node/${SDK_VERSION}`,
      'X-SDK-Version': SDK_VERSION,
    };

    let lastError: Error | undefined;

    for (let attempt = 0; attempt <= this.maxRetries; attempt++) {
      if (attempt > 0) {
        // Exponential backoff: 1s, 2s, 4s
        await new Promise(resolve => setTimeout(resolve, 2 ** (attempt - 1) * 1000));
      }

      try {
        const controller = new AbortController();
        const timeoutId = setTimeout(() => controller.abort(), this.timeout);

        const response = await fetch(url, {
          method,
          headers,
          body: options.body ? JSON.stringify(options.body) : undefined,
          signal: controller.signal,
        });

        clearTimeout(timeoutId);

        const requestId = response.headers.get('x-request-id') ?? 'unknown';

        if (!response.ok) {
          const error = await response.json().catch(() => ({}));

          // Не ретраим клиентские ошибки
          if (response.status < 500) {
            throw new MyApiError(
              error.message ?? 'API Error',
              response.status,
              error.code ?? 'UNKNOWN',
              requestId
            );
          }

          lastError = new MyApiError(
            error.message ?? 'Server Error',
            response.status,
            error.code ?? 'SERVER_ERROR',
            requestId
          );
          continue;
        }

        if (response.status === 204) return undefined as T;
        return response.json();
      } catch (error) {
        if (error instanceof MyApiError) throw error;
        lastError = error as Error;
      }
    }

    throw lastError;
  }
}

Ресурс с пагинацией

// src/resources/projects.ts
export interface Project {
  id: string;
  name: string;
  status: 'active' | 'archived';
  createdAt: string;
}

export interface ListProjectsParams {
  limit?: number;
  cursor?: string;
  status?: 'active' | 'archived';
}

export interface PaginatedResponse<T> {
  data: T[];
  nextCursor?: string;
  hasMore: boolean;
  total: number;
}

export class ProjectsResource {
  constructor(private client: MyApiClient) {}

  async list(params: ListProjectsParams = {}): Promise<PaginatedResponse<Project>> {
    const query = new URLSearchParams();
    if (params.limit) query.set('limit', params.limit.toString());
    if (params.cursor) query.set('cursor', params.cursor);
    if (params.status) query.set('status', params.status);

    return this.client.request('GET', `/projects?${query}`);
  }

  // Auto-paging iterator
  async *listAll(params: Omit<ListProjectsParams, 'cursor'> = {}): AsyncIterable<Project> {
    let cursor: string | undefined;

    do {
      const page = await this.list({ ...params, cursor, limit: params.limit ?? 100 });
      yield* page.data;
      cursor = page.nextCursor;
    } while (cursor);
  }

  async create(data: { name: string; description?: string }): Promise<Project> {
    return this.client.request('POST', '/projects', { body: data });
  }

  async get(id: string): Promise<Project> {
    return this.client.request('GET', `/projects/${id}`);
  }

  async update(id: string, data: Partial<Pick<Project, 'name' | 'status'>>): Promise<Project> {
    return this.client.request('PATCH', `/projects/${id}`, { body: data });
  }

  async delete(id: string): Promise<void> {
    return this.client.request('DELETE', `/projects/${id}`);
  }
}

Верификация webhook

// src/resources/webhooks.ts
import { createHmac, timingSafeEqual } from 'crypto';

export class WebhooksResource {
  constructor(private client: MyApiClient) {}

  verify(payload: string | Buffer, signature: string, secret: string): boolean {
    const expectedSignature = 'sha256=' + createHmac('sha256', secret)
      .update(payload)
      .digest('hex');

    const sigBuffer = Buffer.from(signature);
    const expectedBuffer = Buffer.from(expectedSignature);

    if (sigBuffer.length !== expectedBuffer.length) return false;
    return timingSafeEqual(sigBuffer, expectedBuffer);
  }

  constructEvent(
    payload: string,
    signature: string,
    secret: string
  ): WebhookEvent {
    if (!this.verify(payload, signature, secret)) {
      throw new Error('Invalid webhook signature');
    }
    return JSON.parse(payload);
  }
}

Публикация в npm

{
  "name": "@mycompany/api-sdk",
  "version": "1.0.0",
  "main": "./dist/index.js",
  "module": "./dist/index.mjs",
  "types": "./dist/index.d.ts",
  "exports": {
    ".": {
      "import": "./dist/index.mjs",
      "require": "./dist/index.js",
      "types": "./dist/index.d.ts"
    }
  },
  "files": ["dist"],
  "scripts": {
    "build": "tsup src/index.ts --format cjs,esm --dts",
    "test": "vitest run",
    "prepublishOnly": "npm run build && npm test"
  }
}
// Использование SDK
import { MyApiClient } from '@mycompany/api-sdk';

const api = new MyApiClient({ apiKey: process.env.MYAPI_KEY! });

// Простые операции
const project = await api.projects.create({ name: 'New Project' });

// Auto-paging
for await (const project of api.projects.listAll({ status: 'active' })) {
  console.log(project.name);
}

// Webhook-верификация
const event = api.webhooks.constructEvent(rawBody, signature, webhookSecret);

Разработка TypeScript SDK с авторетраями, пагинацией и публикацией в npm — 3–5 рабочих дней.