1
0
mirror of https://github.com/stevenferrer/multi-select-facet.git synced 2025-11-23 21:54:45 +02:00
Files
multi-select-facet/cmd/api/main.go

353 lines
7.5 KiB
Go
Raw Normal View History

2020-06-09 00:12:27 +08:00
package main
2020-06-09 02:45:17 +08:00
import (
"context"
2020-06-09 19:15:43 +08:00
"encoding/json"
"flag"
"io/ioutil"
2020-06-09 02:45:17 +08:00
"log"
2020-06-09 19:15:43 +08:00
"net/http"
2020-06-09 02:45:17 +08:00
2020-06-09 19:15:43 +08:00
"github.com/go-chi/chi"
"github.com/go-chi/chi/middleware"
"github.com/go-chi/cors"
2020-06-12 21:38:11 +08:00
"github.com/pkg/errors"
2020-06-29 16:01:17 +08:00
solr "github.com/stevenferrer/solr-go"
solrconfig "github.com/stevenferrer/solr-go/config"
2020-06-09 19:15:43 +08:00
solrindex "github.com/stevenferrer/solr-go/index"
solrschema "github.com/stevenferrer/solr-go/schema"
2020-06-09 02:45:17 +08:00
)
2020-06-13 23:21:28 +08:00
// Any is a convenience type for interface{}
type Any = interface{}
// Map is a convenience type for map[string]interface{}
type Map = map[string]Any
2020-06-12 18:50:55 +08:00
2020-06-09 19:15:43 +08:00
const (
2020-06-13 23:21:28 +08:00
dataPath = "products.json"
2020-06-28 18:19:16 +08:00
coll = "multi-select-facet-demo"
2020-06-09 19:15:43 +08:00
)
2020-06-09 02:45:17 +08:00
2020-06-09 19:15:43 +08:00
func main() {
2020-06-28 18:19:16 +08:00
collection := flag.String("collection", coll, "specify the name of collection")
2020-06-29 16:01:17 +08:00
initSchema := flag.Bool("init-schema", false, "initialize solr schema")
initSuggester := flag.Bool("init-suggester", false, "initialize suggester component")
indexData := flag.Bool("index-data", false, "index the products data")
2020-06-09 19:15:43 +08:00
flag.Parse()
2020-06-09 02:45:17 +08:00
2020-06-09 19:15:43 +08:00
solrClient := solr.NewClient("localhost", 8983)
2020-06-09 02:45:17 +08:00
2020-06-12 21:38:11 +08:00
ctx := context.Background()
2020-06-09 19:15:43 +08:00
if *initSchema {
log.Print("initializing solr schema...")
2020-06-29 16:01:17 +08:00
err := initSolrSchema(ctx, *collection, solrClient)
2020-06-09 19:15:43 +08:00
if err != nil {
log.Fatal(err)
}
}
2020-06-29 16:01:17 +08:00
if *indexData {
2020-06-09 19:15:43 +08:00
log.Println("indexing products...")
2020-06-13 23:21:28 +08:00
err := indexProducts(ctx, *collection, solrClient.Index())
2020-06-09 19:15:43 +08:00
if err != nil {
log.Fatal(err)
}
}
2020-06-29 16:01:17 +08:00
if *initSuggester {
log.Println("initializing suggester component...")
err := initSuggestConfig(ctx, *collection, solrClient.Config())
if err != nil {
log.Fatal(err)
}
}
2020-06-09 19:15:43 +08:00
r := chi.NewRouter()
// basic middlewares
r.Use(middleware.Logger)
r.Use(cors.Handler(cors.Options{
AllowedOrigins: []string{"*"},
AllowedMethods: []string{"GET", "POST", "PUT", "DELETE", "OPTIONS"},
AllowedHeaders: []string{"Accept", "Authorization", "Content-Type", "X-CSRF-Token"},
ExposedHeaders: []string{"Link"},
AllowCredentials: false,
MaxAge: 300,
}))
// search handler
r.Method(http.MethodGet, "/search", &searchHandler{
2020-06-13 23:21:28 +08:00
collection: *collection,
2020-06-09 19:15:43 +08:00
solrClient: solrClient,
})
2020-06-26 18:26:10 +08:00
r.Method(http.MethodGet, "/suggest", &suggestHandler{
collection: *collection,
solrClient: solrClient,
})
2020-06-09 19:15:43 +08:00
addr := ":8081"
log.Printf("listening on %s\n", addr)
err := http.ListenAndServe(addr, r)
if err != nil {
log.Fatal(err)
}
2020-06-09 02:45:17 +08:00
}
2020-06-29 16:01:17 +08:00
func initSolrSchema(ctx context.Context, collection string, solrClient solr.Client) (err error) {
2020-06-28 18:19:16 +08:00
// auto-suggest field type
2020-06-26 18:26:10 +08:00
fieldTypes := []solrschema.FieldType{
2020-06-28 18:19:16 +08:00
2020-06-30 16:14:55 +08:00
// // approach #1
// // see: https://blog.griddynamics.com/implementing-autocomplete-with-solr/
// {
2020-06-30 16:14:55 +08:00
// Name: "text_suggest",
// Class: "solr.TextField",
// PositionIncrementGap: "100",
// IndexAnalyzer: &solrschema.Analyzer{
// Tokenizer: &solrschema.Tokenizer{
2020-06-30 16:14:55 +08:00
// Class: "solr.StandardTokenizerFactory",
// },
// Filters: []solrschema.Filter{
// {
// Class: "solr.LowerCaseFilterFactory",
// },
// {
2020-06-30 16:14:55 +08:00
// Class: "solr.EdgeNGramFilterFactory",
// MinGramSize: 1,
// MaxGramSize: 100,
// },
// },
// },
// QueryAnalyzer: &solrschema.Analyzer{
// Tokenizer: &solrschema.Tokenizer{
2020-06-30 16:14:55 +08:00
// Class: "solr.KeywordTokenizerFactory",
// },
// Filters: []solrschema.Filter{
// {
// Class: "solr.LowerCaseFilterFactory",
// },
// },
// },
// },
2020-06-30 16:14:55 +08:00
// approach #2
// see: https://blog.griddynamics.com/implement-autocomplete-search-for-large-e-commerce-catalogs/
{
Name: "text_suggest",
Class: "solr.TextField",
Stored: true,
IndexAnalyzer: &solrschema.Analyzer{
Tokenizer: &solrschema.Tokenizer{
Class: "solr.WhitespaceTokenizerFactory",
},
Filters: []solrschema.Filter{
{
Class: "solr.LowerCaseFilterFactory",
},
{
Class: "solr.ASCIIFoldingFilterFactory",
},
},
},
QueryAnalyzer: &solrschema.Analyzer{
Tokenizer: &solrschema.Tokenizer{
Class: "solr.WhitespaceTokenizerFactory",
},
Filters: []solrschema.Filter{
{
Class: "solr.LowerCaseFilterFactory",
},
{
Class: "solr.ASCIIFoldingFilterFactory",
},
{
Class: "solr.SynonymGraphFilterFactory",
Synonyms: "synonyms.txt",
},
},
},
},
2020-06-26 18:26:10 +08:00
}
for _, fieldType := range fieldTypes {
2020-06-29 16:01:17 +08:00
err = solrClient.Schema().AddFieldType(ctx, collection, fieldType)
2020-06-26 18:26:10 +08:00
if err != nil {
return errors.Wrap(err, "add field type")
}
}
2020-06-12 21:38:11 +08:00
// define the fields
2020-06-09 19:15:43 +08:00
fields := []solrschema.Field{
{
Name: "docType",
Type: "string",
Indexed: true,
Stored: true,
2020-06-09 02:45:17 +08:00
},
2020-06-09 19:15:43 +08:00
{
2020-06-12 18:50:55 +08:00
Name: "name",
2020-06-09 19:15:43 +08:00
Type: "text_general",
Indexed: true,
Stored: true,
2020-06-09 02:45:17 +08:00
},
2020-06-09 19:15:43 +08:00
{
2020-06-12 18:50:55 +08:00
Name: "category",
Type: "text_gen_sort",
2020-06-09 19:15:43 +08:00
Indexed: true,
Stored: true,
},
{
Name: "brand",
Type: "text_gen_sort",
Indexed: true,
Stored: true,
},
{
2020-06-12 18:50:55 +08:00
Name: "productType",
2020-06-09 19:15:43 +08:00
Type: "string",
Indexed: true,
Stored: true,
2020-06-09 02:45:17 +08:00
},
2020-06-26 18:26:10 +08:00
{
2020-06-28 18:19:16 +08:00
Name: "suggest",
Type: "text_suggest",
Stored: true,
MultiValued: true,
2020-06-26 18:26:10 +08:00
},
2020-06-09 02:45:17 +08:00
}
2020-06-09 19:15:43 +08:00
for _, field := range fields {
2020-06-29 16:01:17 +08:00
err = solrClient.Schema().AddField(ctx, collection, field)
2020-06-09 19:15:43 +08:00
if err != nil {
2020-06-12 21:38:11 +08:00
return errors.Wrap(err, "add field")
2020-06-09 02:45:17 +08:00
}
2020-06-09 19:15:43 +08:00
}
2020-06-09 02:45:17 +08:00
2020-06-26 18:26:10 +08:00
copyFields := []solrschema.CopyField{
{
Source: "name",
Dest: "suggest",
},
2020-06-28 18:19:16 +08:00
{
Source: "category",
Dest: "suggest",
},
{
Source: "brand",
Dest: "suggest",
},
{
Source: "productType",
Dest: "suggest",
},
{
Source: "*_s",
Dest: "suggest",
},
{
Source: "name",
Dest: "_text_",
},
{
Source: "category",
Dest: "_text_",
},
{
Source: "brand",
Dest: "_text_",
},
{
Source: "productType",
Dest: "_text_",
},
{
Source: "*_s",
Dest: "_text_",
},
2020-06-26 18:26:10 +08:00
}
for _, copyField := range copyFields {
2020-06-29 16:01:17 +08:00
err = solrClient.Schema().AddCopyField(ctx, collection, copyField)
2020-06-26 18:26:10 +08:00
if err != nil {
return errors.Wrap(err, "add copy field")
}
}
2020-06-09 19:15:43 +08:00
return nil
}
2020-06-09 02:45:17 +08:00
2020-06-29 16:01:17 +08:00
func initSuggestConfig(ctx context.Context, collection string, configClient solrconfig.Client) error {
// suggester configs
addSuggestComponent := solrconfig.NewComponentCommand(
2020-07-05 21:05:14 +08:00
solrconfig.AddSearchComponent, Map{
2020-06-29 16:01:17 +08:00
"name": "suggest",
"class": "solr.SuggestComponent",
2020-07-05 21:05:14 +08:00
"suggester": Map{
"name": "default",
2020-06-29 16:01:17 +08:00
"lookupImpl": "FuzzyLookupFactory",
"dictionaryImpl": "DocumentDictionaryFactory",
"field": "suggest",
"suggestAnalyzerFieldType": "text_suggest",
},
},
)
addSuggestHandler := solrconfig.NewComponentCommand(
2020-07-05 21:05:14 +08:00
solrconfig.AddRequestHandler, Map{
2020-06-29 16:01:17 +08:00
"name": "/suggest",
"class": "solr.SearchHandler",
"startup": "lazy",
2020-07-05 21:05:14 +08:00
"defaults": Map{
2020-06-29 16:01:17 +08:00
"suggest": true,
"suggest.count": 10,
"suggest.dictionary": "default",
2020-06-29 16:01:17 +08:00
},
"components": []string{"suggest"},
},
)
err := configClient.SendCommands(ctx, collection,
addSuggestComponent, addSuggestHandler)
if err != nil {
return errors.Wrap(err, "add suggester configs")
}
return nil
}
2020-06-13 23:21:28 +08:00
func indexProducts(ctx context.Context, collection string,
indexClient solrindex.Client) error {
2020-06-12 18:50:55 +08:00
b, err := ioutil.ReadFile(dataPath)
2020-06-09 19:15:43 +08:00
if err != nil {
return err
}
2020-06-09 02:45:17 +08:00
2020-06-26 18:26:10 +08:00
var products []Map
err = json.Unmarshal(b, &products)
2020-06-09 19:15:43 +08:00
if err != nil {
return err
}
2020-06-26 18:26:10 +08:00
docs := solrindex.NewDocs()
for _, product := range products {
docs.AddDoc(product)
}
err = indexClient.AddDocuments(ctx, collection, docs)
2020-06-13 23:21:28 +08:00
if err != nil {
return err
}
// commit updates
err = indexClient.Commit(ctx, collection)
2020-06-09 19:15:43 +08:00
if err != nil {
return err
2020-06-09 02:45:17 +08:00
}
2020-06-09 00:12:27 +08:00
2020-06-09 19:15:43 +08:00
return nil
2020-06-09 02:45:17 +08:00
}