Building a semantic search API in Go with Meilisearch This article provides a tutorial on building a semantic search API in Go using the Fiber web framework and Meilisearch, a typo-tolerant search engine. The architecture includes filter support, configurable typo tolerance, and a MySQL LIKE query fallback for resilience, and is used to power search across over 1,600 cybersecurity articles. The guide covers project setup, index configuration with searchable and filterable attributes, and document indexing. Full-text search is one of those features that looks simple until you have to ship it. Typos fail silently. Category filters conflict with relevance ranking. The database LIKE query that worked at 10,000 rows grinds to a halt at 100,000. This tutorial walks through building a real search API in Go using Fiber and Meilisearch, complete with filter support, typo tolerance configuration, and a MySQL LIKE fallback for resilience. This is roughly the architecture running search across 1,600+ cybersecurity articles at AYI NEDJIMI Consultants https://ayinedjimi-consultants.fr/articles . Setup go mod init search-api go get github.com/gofiber/fiber/v2 go get github.com/meilisearch/meilisearch-go go get github.com/go-sql-driver/mysql Run Meilisearch: docker run -d -p 7700:7700 \ -e MEILI MASTER KEY=your master key \ getmeili/meilisearch:latest Project structure search-api/ ├── main.go ├── config/ │ └── config.go ├── search/ │ ├── meili.go │ └── fallback.go └── handlers/ └── search.go Data model // search/meili.go package search // Article is the document type stored in Meilisearch and MySQL. type Article struct { ID string json:"id" Title string json:"title" Slug string json:"slug" Content string json:"content" // plain text, stripped of HTML Category string json:"category" // news, guide, analyse, blog, checklist Difficulty string json:"difficulty" // beginner, intermediate, advanced DocType string json:"doc type" // article, checklist, glossary Tags string json:"tags" PublishedAt int64 json:"published at" // Unix timestamp for sort } // SearchResult wraps hits with metadata. type SearchResult struct { Hits Article json:"hits" TotalHits int64 json:"total hits" ProcessingTimeMs int64 json:"processing time ms" Query string json:"query" Source string json:"source" // "meilisearch" or "mysql fallback" } Meilisearch client initialization // search/meili.go continued package search import "fmt" "log" "github.com/meilisearch/meilisearch-go" const IndexName = "articles" type MeiliSearcher struct { client meilisearch.ServiceManager index meilisearch.IndexManager } func NewMeiliSearcher host, apiKey string MeiliSearcher, error { client := meilisearch.New host, meilisearch.WithAPIKey apiKey // Verify connectivity if , err := client.Health ; err = nil { return nil, fmt.Errorf "meilisearch unreachable at %s: %w", host, err } s := &MeiliSearcher{client: client} if err := s.ensureIndex ; err = nil { return nil, err } return s, nil } func s MeiliSearcher ensureIndex error { // Get or create index idx, err := s.client.GetIndex IndexName if err = nil { task, err := s.client.CreateIndex &meilisearch.IndexConfig{ Uid: IndexName, PrimaryKey: "id", } if err = nil { return fmt.Errorf "create index: %w", err } s.client.WaitForTask task.TaskUID, nil idx, err = s.client.GetIndex IndexName if err = nil { return fmt.Errorf "get index after create: %w", err } } s.index = idx return s.configureIndex } func s MeiliSearcher configureIndex error { task, err := s.index.UpdateSettings &meilisearch.Settings{ SearchableAttributes: string{ "title", // highest weight "tags", "content", // lowest weight }, FilterableAttributes: string{ "category", "difficulty", "doc type", }, SortableAttributes: string{ "published at", }, RankingRules: string{ "words", "typo", "proximity", "attribute", // respects SearchableAttributes order "sort", "exactness", }, TypoTolerance: &meilisearch.TypoTolerance{ Enabled: func bool { b := true; return &b } , MinWordSizeForTypos: meilisearch.MinWordSizeForTypos{ OneTypo: 4, TwoTypos: 8, }, }, Pagination: &meilisearch.Pagination{ MaxTotalHits: 10000, }, } if err = nil { return fmt.Errorf "update settings: %w", err } s.client.WaitForTask task.TaskUID, nil log.Println "Meilisearch index configured." return nil } Index sync function Call this on startup and hook it to your CRUD operations. // search/meili.go continued func s MeiliSearcher IndexDocuments articles Article error { if len articles == 0 { return nil } task, err := s.index.AddDocuments articles, "id" if err = nil { return fmt.Errorf "add documents: %w", err } s.client.WaitForTask task.TaskUID, nil log.Printf "Indexed %d documents.", len articles return nil } func s MeiliSearcher DeleteDocument id string error { task, err := s.index.DeleteDocument id if err = nil { return fmt.Errorf "delete document %s: %w", id, err } s.client.WaitForTask task.TaskUID, nil return nil } Search with filters // search/meili.go continued type SearchParams struct { Query string Category string Difficulty string DocType string Limit int64 Offset int64 } func s MeiliSearcher Search params SearchParams SearchResult, error { if params.Limit <= 0 { params.Limit = 10 } if params.Limit 100 { params.Limit = 100 } // Build filter expression var filters string if params.Category = "" { filters = append filters, fmt.Sprintf "category = %q", params.Category } if params.Difficulty = "" { filters = append filters, fmt.Sprintf "difficulty = %q", params.Difficulty } if params.DocType = "" { filters = append filters, fmt.Sprintf "doc type = %q", params.DocType } filterStr := "" for i, f := range filters { if i == 0 { filterStr = f } else { filterStr += " AND " + f } } req := &meilisearch.SearchRequest{ Limit: params.Limit, Offset: params.Offset, AttributesToRetrieve: string{ "id", "title", "slug", "category", "difficulty", "doc type", "tags", "published at", }, // Don't return full content in search results } if filterStr = "" { req.Filter = filterStr } resp, err := s.index.Search params.Query, req if err = nil { return nil, err } hits := make Article, 0, len resp.Hits for , h := range resp.Hits { // Meilisearch returns hits as map string interface{} b, := json.Marshal h var a Article if err := json.Unmarshal b, &a ; err == nil { hits = append hits, a } } return &SearchResult{ Hits: hits, TotalHits: resp.TotalHits, ProcessingTimeMs: resp.ProcessingTimeMs, Query: params.Query, Source: "meilisearch", }, nil } MySQL fallback When Meilisearch is down restart, OOM, maintenance , fall back to MySQL LIKE. It's slower and has no typo tolerance, but it keeps the API responding. // search/fallback.go package search import "database/sql" "fmt" "strings" "time" "github.com/go-sql-driver/mysql" type MySQLFallback struct { db sql.DB } func NewMySQLFallback dsn string MySQLFallback, error { db, err := sql.Open "mysql", dsn if err = nil { return nil, err } db.SetMaxOpenConns 10 db.SetConnMaxLifetime 5 time.Minute return &MySQLFallback{db: db}, nil } func m MySQLFallback Search params SearchParams SearchResult, error { query := "%" + strings.ReplaceAll params.Query, "%", "\\%" + "%" args := interface{}{query, query} where := "WHERE title LIKE ? OR content LIKE ? " if params.Category = "" { where += " AND category = ?" args = append args, params.Category } if params.Difficulty = "" { where += " AND difficulty = ?" args = append args, params.Difficulty } if params.DocType = "" { where += " AND doc type = ?" args = append args, params.DocType } // Count total var total int64 countSQL := fmt.Sprintf "SELECT COUNT FROM articles %s", where = m.db.QueryRow countSQL, args... .Scan &total // Fetch page args = append args, params.Limit, params.Offset rows, err := m.db.Query fmt.Sprintf SELECT id, title, slug, category, difficulty, doc type, published at FROM articles %s ORDER BY published at DESC LIMIT ? OFFSET ? , where , args..., if err = nil { return nil, err } defer rows.Close var hits Article for rows.Next { var a Article if err := rows.Scan &a.ID, &a.Title, &a.Slug, &a.Category, &a.Difficulty, &a.DocType, &a.PublishedAt ; err = nil { continue } hits = append hits, a } return &SearchResult{ Hits: hits, TotalHits: total, Query: params.Query, Source: "mysql fallback", }, nil } HTTP handler // handlers/search.go package handlers import "encoding/json" "log" "github.com/gofiber/fiber/v2" "search-api/search" type SearchHandler struct { meili search.MeiliSearcher fallback search.MySQLFallback } func NewSearchHandler meili search.MeiliSearcher, fallback search.MySQLFallback SearchHandler { return &SearchHandler{meili: meili, fallback: fallback} } func h SearchHandler Handle c fiber.Ctx error { params := search.SearchParams{ Query: c.Query "q", "" , Category: c.Query "cat", "" , Difficulty: c.Query "diff", "" , DocType: c.Query "type", "" , Limit: int64 c.QueryInt "limit", 10 , Offset: int64 c.QueryInt "offset", 0 , } if len rune params.Query 200 { return c.Status fiber.StatusBadRequest .JSON fiber.Map{ "error": "query too long", } } result, err := h.meili.Search params if err = nil { log.Printf "Meilisearch error: %v — falling back to MySQL", err result, err = h.fallback.Search params if err = nil { return c.Status fiber.StatusInternalServerError .JSON fiber.Map{ "error": "search unavailable", } } } return c.JSON result } Wiring it up in main.go // main.go package main import "log" "os" "github.com/gofiber/fiber/v2" "search-api/handlers" "search-api/search" func main { meiliHost := os.Getenv "MEILI HOST" // e.g. "http://127.0.0.1:7700" meiliKey := os.Getenv "MEILI MASTER KEY" mysqlDSN := os.Getenv "MYSQL DSN" // e.g. "user:pass@tcp 127.0.0.1:3306 /dbname" meili, err := search.NewMeiliSearcher meiliHost, meiliKey if err = nil { log.Fatalf "Failed to connect to Meilisearch: %v", err } fallback, err := search.NewMySQLFallback mysqlDSN if err = nil { log.Fatalf "Failed to connect to MySQL: %v", err } app := fiber.New fiber.Config{ ErrorHandler: func c fiber.Ctx, err error error { return c.Status fiber.StatusInternalServerError .JSON fiber.Map{ "error": err.Error , } }, } searchHandler := handlers.NewSearchHandler meili, fallback app.Get "/api/search", searchHandler.Handle // Reindex endpoint protect with auth middleware in production app.Post "/admin/search/reindex", func c fiber.Ctx error { var articles search.Article if err := c.BodyParser &articles ; err = nil { return c.Status fiber.StatusBadRequest .JSON fiber.Map{"error": err.Error } } if err := meili.IndexDocuments articles ; err = nil { return c.Status fiber.StatusInternalServerError .JSON fiber.Map{"error": err.Error } } return c.JSON fiber.Map{"indexed": len articles } } log.Fatal app.Listen ":4001" } Test it Basic search curl "http://localhost:4001/api/search?q=pentest" With filters curl "http://localhost:4001/api/search?q=NIS+2&cat=guide&diff=intermediate&limit=5" Typo tolerance in action curl "http://localhost:4001/api/search?q=penetartion+tesitng" Key tuning decisions SearchableAttributes order matters. Meilisearch's attribute ranking rule rewards matches in earlier attributes. Putting title first means a title match outranks a content match, which is almost always what you want. Pagination cap. The MaxTotalHits: 10000 setting prevents Meilisearch from doing expensive full-index scans for pagination deep into results. If users never go past page 20 at 10 results/page, set this to 200. Fallback opacity. The source field in SearchResult tells the frontend and your monitoring when it's getting degraded results. Log every fallback occurrence — if Meilisearch is down and you're not paged, you won't know until users complain. On-startup sync. Pull all published articles from MySQL on startup and call IndexDocuments . For 10,000+ documents this takes 2–5 seconds; for 100,000+ batch in chunks of 5,000. This ensures your index is always consistent even after a Meilisearch restart. This pattern — Meilisearch primary, MySQL LIKE fallback — gives you production-grade search with no single point of failure and a codebase any Go developer can reason about.