Реалізація бази знань та FAQ на сайті

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

Це лише деякі з технічних типів сайтів, з якими ми працюємо, і кожен із них може мати свої специфічні особливості та функціональність, а також бути адаптованим під конкретні потреби та цілі клієнта.

Пропоновані послуги
Показано 1 з 1 послугУсі 2065 послуг
Реалізація бази знань та FAQ на сайті
Середня
~1-2 тижні
Часті питання
Наші компетенції:
Етапи розробки
Останні роботи
  • 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

Реалізація бази знань та FAQ

База знань — структурований каталог статей з пошуком. FAQ — плоский список питання/відповідь. Обидві задачі вирішуються подібним стеком, різниця — в ієрархії категорій та повнотекстовому пошуку.

Структура бази даних

CREATE TABLE kb_categories (
    id        SERIAL PRIMARY KEY,
    parent_id INTEGER REFERENCES kb_categories(id),
    name      VARCHAR(150) NOT NULL,
    slug      VARCHAR(150) NOT NULL UNIQUE,
    icon      VARCHAR(50),
    sort_order INTEGER NOT NULL DEFAULT 0
);

CREATE TABLE kb_articles (
    id          SERIAL PRIMARY KEY,
    category_id INTEGER REFERENCES kb_categories(id),
    title       VARCHAR(255) NOT NULL,
    slug        VARCHAR(255) NOT NULL UNIQUE,
    excerpt     TEXT,
    body        TEXT NOT NULL,
    body_search TSVECTOR GENERATED ALWAYS AS (
        to_tsvector('russian', title || ' ' || body)
    ) STORED,
    helpful_yes  INTEGER NOT NULL DEFAULT 0,
    helpful_no   INTEGER NOT NULL DEFAULT 0,
    views_count  INTEGER NOT NULL DEFAULT 0,
    is_published BOOLEAN NOT NULL DEFAULT true,
    sort_order   INTEGER NOT NULL DEFAULT 0,
    created_at  TIMESTAMPTZ NOT NULL DEFAULT NOW()
);

CREATE INDEX ON kb_articles USING gin(body_search);
CREATE INDEX ON kb_articles(category_id, is_published, sort_order);

CREATE TABLE faq_items (
    id       SERIAL PRIMARY KEY,
    category VARCHAR(100),
    question TEXT NOT NULL,
    answer   TEXT NOT NULL,
    sort_order INTEGER NOT NULL DEFAULT 0
);

Laravel: пошук та API

class KnowledgeBaseController extends Controller
{
    // Повнотекстовий пошук
    public function search(Request $request): JsonResponse
    {
        $query = trim($request->input('q', ''));

        if (strlen($query) < 2) {
            return response()->json(['data' => [], 'query' => $query]);
        }

        $articles = KbArticle::published()
            ->whereRaw(
                "body_search @@ plainto_tsquery('russian', ?)",
                [$query]
            )
            ->selectRaw("*, ts_rank(body_search, plainto_tsquery('russian', ?)) as rank", [$query])
            ->orderByDesc('rank')
            ->limit(10)
            ->get(['id', 'title', 'slug', 'excerpt', 'category_id', 'rank']);

        return response()->json([
            'data'  => KbArticleResource::collection($articles),
            'query' => $query,
        ]);
    }

    // Стаття з трекінгом переглядів
    public function show(string $slug): JsonResponse
    {
        $article = KbArticle::published()
            ->with('category')
            ->where('slug', $slug)
            ->firstOrFail();

        // Інкрементувати перегляди (асинхронно)
        dispatch(fn() => $article->increment('views_count'))->afterResponse();

        // Пов'язані статті тієї ж категорії
        $related = KbArticle::published()
            ->where('category_id', $article->category_id)
            ->where('id', '!=', $article->id)
            ->orderByDesc('views_count')
            ->limit(5)
            ->get(['id', 'title', 'slug']);

        return response()->json([
            'article' => KbArticleResource::make($article),
            'related' => $related,
        ]);
    }

    // Оцінити корисність статті
    public function helpful(Request $request, KbArticle $article): JsonResponse
    {
        $request->validate(['helpful' => 'required|boolean']);

        $session = $request->session()->getId();
        $key = "helpful:{$article->id}:{$session}";

        if (Cache::has($key)) {
            return response()->json(['already_voted' => true]);
        }

        Cache::put($key, true, now()->addDays(30));

        if ($request->boolean('helpful')) {
            $article->increment('helpful_yes');
        } else {
            $article->increment('helpful_no');
        }

        return response()->json([
            'yes' => $article->fresh()->helpful_yes,
            'no'  => $article->fresh()->helpful_no,
        ]);
    }
}

React: акордеон FAQ

import { useState } from 'react';

interface FaqItem {
  id: number;
  question: string;
  answer: string;
}

function FaqAccordion({ items, category }: { items: FaqItem[]; category: string }) {
  const [openId, setOpenId] = useState<number | null>(null);

  return (
    <section>
      <h2>{category}</h2>
      <dl>
        {items.map(item => (
          <div key={item.id} className={`faq-item ${openId === item.id ? 'open' : ''}`}>
            <dt>
              <button
                onClick={() => setOpenId(openId === item.id ? null : item.id)}
                aria-expanded={openId === item.id}
                aria-controls={`faq-answer-${item.id}`}
              >
                {item.question}
                <span aria-hidden>{openId === item.id ? '−' : '+'}</span>
              </button>
            </dt>
            <dd
              id={`faq-answer-${item.id}`}
              hidden={openId !== item.id}
            >
              <div dangerouslySetInnerHTML={{ __html: item.answer }} />
            </dd>
          </div>
        ))}
      </dl>
    </section>
  );
}

React: пошук по базі знань

function KbSearch() {
  const [query, setQuery] = useState('');
  const [results, setResults] = useState<KbArticle[]>([]);

  useEffect(() => {
    if (query.length < 2) { setResults([]); return; }

    const timer = setTimeout(async () => {
      const { data } = await api.get('/api/kb/search', { params: { q: query } });
      setResults(data.data);
    }, 300);

    return () => clearTimeout(timer);
  }, [query]);

  return (
    <div className="kb-search">
      <input
        type="search"
        value={query}
        onChange={e => setQuery(e.target.value)}
        placeholder="Пошук по базі знань..."
        aria-label="Пошук по базі знань"
      />
      {results.length > 0 && (
        <ul className="kb-search__results" role="listbox">
          {results.map(article => (
            <li key={article.id} role="option">
              <a href={`/help/${article.slug}`}>
                <strong>{article.title}</strong>
                <p>{article.excerpt}</p>
              </a>
            </li>
          ))}
        </ul>
      )}
    </div>
  );
}

Розмітка FAQ Schema.org

// У Blade-шаблоні
<script type="application/ld+json">
{
  "@context": "https://schema.org",
  "@type": "FAQPage",
  "mainEntity": [
    @foreach($faqItems as $item)
    {
      "@type": "Question",
      "name": {{ json_encode($item->question) }},
      "acceptedAnswer": {
        "@type": "Answer",
        "text": {{ json_encode(strip_tags($item->answer)) }}
      }
    }{{ $loop->last ? '' : ',' }}
    @endforeach
  ]
}
</script>

Розмітка FAQ Schema.org дозволяє FAQ з'являтися в розширених результатах пошуку Google.

Терміни реалізації

База знань з категоріями та PostgreSQL повнотекстовим пошуком: 3–4 дні. З FAQ-акордеоном, розміткою Schema.org та оцінкою корисності: 4–5 днів.