golang操作Elasticsearch

MichaelCc發表於2018-12-02

go操作Elasticsearch主要有以下兩個sdk:

  • http://github.com/olivere/elastic 第三方開發,各個版本都有對應的sdk,文件也豐富
  • http://github.com/elastic/go-elasticsearch

 最終我們選擇了olivere/elastic包

操作程式碼如下:

package main
 
import (
    "context"
    "encoding/json"
    "fmt"
    "gopkg.in/olivere/elastic.v5" //這裡使用的是版本5,最新的是6,有改動
    "log"
    "os"
    "reflect"
)
 
var client *elastic.Client
var host = "http://127.0.0.1:9200/"
 
type Employee struct {
    FirstName string   `json:"first_name"`
    LastName  string   `json:"last_name"`
    Age       int      `json:"age"`
    About     string   `json:"about"`
    Interests []string `json:"interests"`
}
 
//初始化
func init() {
    errorlog := log.New(os.Stdout, "APP", log.LstdFlags)
    var err error
    client, err = elastic.NewClient(elastic.SetErrorLog(errorlog), elastic.SetURL(host))
    if err != nil {
        panic(err)
    }
    info, code, err := client.Ping(host).Do(context.Background())
    if err != nil {
        panic(err)
    }
    fmt.Printf("Elasticsearch returned with code %d and version %s\n", code, info.Version.Number)
 
    esversion, err := client.ElasticsearchVersion(host)
    if err != nil {
        panic(err)
    }
    fmt.Printf("Elasticsearch version %s\n", esversion)
 
}
 
/*下面是簡單的CURD*/
 
//建立
func create() {
 
    //使用結構體
    e1 := Employee{"Jane", "Smith", 32, "I like to collect rock albums", []string{"music"}}
    put1, err := client.Index().
        Index("megacorp").
        Type("employee").
        Id("1").
        BodyJson(e1).
        Do(context.Background())
    if err != nil {
        panic(err)
    }
    fmt.Printf("Indexed tweet %s to index s%s, type %s\n", put1.Id, put1.Index, put1.Type)
 
    //使用字串
    e2 := `{"first_name":"John","last_name":"Smith","age":25,"about":"I love to go rock climbing","interests":["sports","music"]}`
    put2, err := client.Index().
        Index("megacorp").
        Type("employee").
        Id("2").
        BodyJson(e2).
        Do(context.Background())
    if err != nil {
        panic(err)
    }
    fmt.Printf("Indexed tweet %s to index s%s, type %s\n", put2.Id, put2.Index, put2.Type)
 
    e3 := `{"first_name":"Douglas","last_name":"Fir","age":35,"about":"I like to build cabinets","interests":["forestry"]}`
    put3, err := client.Index().
        Index("megacorp").
        Type("employee").
        Id("3").
        BodyJson(e3).
        Do(context.Background())
    if err != nil {
        panic(err)
    }
    fmt.Printf("Indexed tweet %s to index s%s, type %s\n", put3.Id, put3.Index, put3.Type)
 
}
 
//刪除
func delete() {
 
    res, err := client.Delete().Index("megacorp").
        Type("employee").
        Id("1").
        Do(context.Background())
    if err != nil {
        println(err.Error())
        return
    }
    fmt.Printf("delete result %s\n", res.Result)
}
 
//修改
func update() {
    res, err := client.Update().
        Index("megacorp").
        Type("employee").
        Id("2").
        Doc(map[string]interface{}{"age": 88}).
        Do(context.Background())
    if err != nil {
        println(err.Error())
    }
    fmt.Printf("update age %s\n", res.Result)
 
}
 
//查詢
func gets() {
    //通過id查詢
    get1, err := client.Get().Index("megacorp").Type("employee").Id("2").Do(context.Background())
    if err != nil {
        panic(err)
    }
    if get1.Found {
        fmt.Printf("Got document %s in version %d from index %s, type %s\n", get1.Id, get1.Version, get1.Index, get1.Type)
    }
}
 
//搜尋
func query() {
    var res *elastic.SearchResult
    var err error
    //取所有
    res, err = client.Search("megacorp").Type("employee").Do(context.Background())
    printEmployee(res, err)
 
    //欄位相等
    q := elastic.NewQueryStringQuery("last_name:Smith")
    res, err = client.Search("megacorp").Type("employee").Query(q).Do(context.Background())
    if err != nil {
        println(err.Error())
    }
    printEmployee(res, err)
 
    if res.Hits.TotalHits > 0 {
        fmt.Printf("Found a total of %d Employee \n", res.Hits.TotalHits)
 
        for _, hit := range res.Hits.Hits {
 
            var t Employee
            err := json.Unmarshal(*hit.Source, &t) //另外一種取資料的方法
            if err != nil {
                fmt.Println("Deserialization failed")
            }
 
            fmt.Printf("Employee name %s : %s\n", t.FirstName, t.LastName)
        }
    } else {
        fmt.Printf("Found no Employee \n")
    }
 
    //條件查詢
    //年齡大於30歲的
    boolQ := elastic.NewBoolQuery()
    boolQ.Must(elastic.NewMatchQuery("last_name", "smith"))
    boolQ.Filter(elastic.NewRangeQuery("age").Gt(30))
    res, err = client.Search("megacorp").Type("employee").Query(q).Do(context.Background())
    printEmployee(res, err)
 
    //短語搜尋 搜尋about欄位中有 rock climbing
    matchPhraseQuery := elastic.NewMatchPhraseQuery("about", "rock climbing")
    res, err = client.Search("megacorp").Type("employee").Query(matchPhraseQuery).Do(context.Background())
    printEmployee(res, err)
 
    //分析 interests
    aggs := elastic.NewTermsAggregation().Field("interests")
    res, err = client.Search("megacorp").Type("employee").Aggregation("all_interests", aggs).Do(context.Background())
    printEmployee(res, err)
 
}
 
//簡單分頁
func list(size,page int) {
    if size < 0 || page < 1 {
        fmt.Printf("param error")
        return
    }
    res,err := client.Search("megacorp").
        Type("employee").
        Size(size).
        From((page-1)*size).
        Do(context.Background())
        printEmployee(res, err)
 
}
 
//列印查詢到的Employee
func printEmployee(res *elastic.SearchResult, err error) {
    if err != nil {
        print(err.Error())
        return
    }
    var typ Employee
    for _, item := range res.Each(reflect.TypeOf(typ)) { //從搜尋結果中取資料的方法
        t := item.(Employee)
        fmt.Printf("%#v\n", t)
    }
}
 
func main() {
    create()
    delete()
    update()
    gets()
    query()
        list()
}
複製程式碼


相關文章