Elasticsearch 是一个分布式的、面向生产规模工作负载优化的搜索引擎。
Kibana 可以将 Elasticsearch 中的数据转化为直观的图表、图形和仪表盘。
这篇文章,您将学习本地安装 Elasticsearch 和 Kibana,以及使用开发工具/ Java SDK 创建索引和搜索数据。
1 本地安装
1.1 创建网络
我们需要创建一个供 Elasticsearch 和 Kibana 使用的 network。这个 network 将被用于 Elasticsearch 和 Kibana 之间的通信。
1
docker network create elastic
1.2 安装 ES
拉取 Elasticsearch 镜像
1
docker pull docker.elastic.co/elasticsearch/elasticsearch:{version}
这里的版本 version ,我们选取:8.9.0。
1
docker pull docker.elastic.co/elasticsearch/elasticsearch:8.9.0
启动 docker elasticsearch 镜像
1
docker run --name elasticsearch --net elastic -p 9200:9200 -p 9300:9300 -e "discovery.type=single-node" -t docker.elastic.co/elasticsearch/elasticsearch:8.9.0
修改 elasticsearch 用户密码
1
bin/elasticsearch-reset-password --username elastic -i
1.3 安装 Kibana
拉取 Kibana 镜像
1
docker pull docker.elastic.co/kibana/kibana:8.9.0
启动 Kibana 镜像
1
docker run --name kibana --net elastic -p 5601:5601 docker.elastic.co/kibana/kibana:8.9.0
因为启动安装 Kibana ,需要 token , 所以进入 elasticsearch 容器 ,执行:
1
bin/elasticsearch-create-enrollment-token -s kibana
输入 token 之后,刷新页面,进入登录页面:
2 接口测试
我们可以使用 Kibana Dev tools 来进行接口测试。
2.1 添加
向索引添加单个文档,提交一个 HTTP POST 请求,目标是该索引。
1
POST /customer/_doc/1
2
{
3
"firstname": "Jennifer",
4
"lastname": "Walters"
5
}
该请求会自动创建名为customer的索引(如果不存在),然后添加一个 ID 为 1的新文档,同时存储并建立firstname和lastname字段的索引。
新文档可以立即从集群中的任何节点获取。您可以使用 GET 请求来检索它,请求中需指定其文档 ID :
1
GET /customer/_doc/1
要一次性添加多个文档,请使用 _bulk API。批量数据必须是以换行分隔的 JSON(NDJSON)格式。每一行必须以换行字符(\n)结尾,包括最后一行。
1
PUT customer/_bulk
2
{ "create": { } }
3
{ "firstname": "Monica","lastname":"Rambeau"}
4
{ "create": { } }
5
{ "firstname": "Carol","lastname":"Danvers"}
6
{ "create": { } }
7
{ "firstname": "Wanda","lastname":"Maximoff"}
8
{ "create": { } }
9
{ "firstname": "Jennifer","lastname":"Takeda"}
2.2 搜索
已索引的文档可以在准实时的情况下进行搜索。下面的搜索将在customer索引中匹配所有名为 Jennifer 的顾客。
1
GET customer/_search
2
{
3
"query" : {
4
"match" : { "firstname": "Jennifer" }
5
}
6
}
2.3 视图
进入 Kibana Data Views :
然后创建数据视图 :
创建数据视图之后,可以在 Analytics > Discover 查看索引数据。
3 Java SDK 实战
3.1 依赖
1
<dependency>
2
<groupId>co.elastic.clients</groupId>
3
<artifactId>elasticsearch-java</artifactId>
4
<version>8.9.0</version>
5
</dependency>
6
7
<dependency>
8
<groupId>io.github.hakky54</groupId>
9
<artifactId>sslcontext-kickstart</artifactId>
10
<version>7.1.0</version>
11
</dependency>
12
13
<dependency>
14
<groupId>com.fasterxml.jackson.core</groupId>
15
<artifactId>jackson-databind</artifactId>
16
<version>2.12.3</version>
17
</dependency>
18
19
<dependency>
20
<groupId>com.fasterxml.jackson.core</groupId>
21
<artifactId>jackson-core</artifactId>
22
<version>2.12.3</version>
23
</dependency>
24
25
<dependency>
26
<groupId>com.fasterxml.jackson.core</groupId>
27
<artifactId>jackson-annotations</artifactId>
28
<version>2.12.3</version>
29
</dependency>
30
31
<dependency>
32
<groupId>jakarta.json</groupId>
33
<artifactId>jakarta.json-api</artifactId>
34
<version>2.0.1</version>
35
</dependency>
36
37
<!-- 强制走高版本 ,防止和springboot 依赖冲突 -->
38
<dependency>
39
<groupId>org.elasticsearch.client</groupId>
40
<artifactId>elasticsearch-rest-client</artifactId>
41
<version>8.9.0</version>
42
</dependency>
3.2 创建客户端
1
RestClientBuilder builder = RestClient.builder(new HttpHost("localhost", 9200, "https"));
2
3
final CredentialsProvider credentialsProvider = new BasicCredentialsProvider();
4
credentialsProvider.setCredentials(AuthScope.ANY, new UsernamePasswordCredentials("elastic", "ilxw@19841201"));
5
6
// Elasticsearch 提供了 Https 服务,创建 client 建立 SSL 链接时没有做证书验证 ;
7
SSLFactory sslFactory = SSLFactory.builder().withUnsafeTrustMaterial().withUnsafeHostnameVerifier().build();
8
builder = builder.setHttpClientConfigCallback(
9
httpClientBuilder -> httpClientBuilder.setDefaultCredentialsProvider(credentialsProvider)
10
setSSLContext(sslFactory.getSslContext())
11
.setSSLHostnameVerifier(sslFactory.getHostnameVerifier())
12
);
13
14
RestClient restClient = builder.build();
15
16
// Create the transport with a Jackson mapper
17
ElasticsearchTransport transport = new RestClientTransport(restClient, new JacksonJsonpMapper());
18
19
// And create the API client
20
ElasticsearchClient esClient = new ElasticsearchClient(transport);
3.3 创建文档
创建索引名为 products ,新建一个文档 id 为 1 。
1
ProductPo product = new ProductPo(1, "Bag", 42);
2
3
IndexRequest<Object> indexRequest = new IndexRequest.Builder<>().index("products").id(String.valueOf(product.getId())).document(product).build();
4
5
IndexResponse response = esClient.index(indexRequest);
6
7
System.out.println("Indexed with version " + response.version());
3.4 查询文档
1
GetResponse<ProductPo> response = esClient.get(g -> g
2
.index("products")
3
.id(String.valueOf(1)),
4
ProductPo.class
5
);
6
7
if (response.found()) {
8
ProductPo product = response.source();
9
System.out.println("Product name " + product.getName());
10
} else {
11
System.out.println("Product not found");
12
}
3.5 修改文档
1
Map<String, Object> doc = new HashMap<String, Object>();
2
//文档产品名称调整为 my bike
3
doc.put("name", "my bike");
4
doc.put("price", 100);
5
6
BulkOperation op = new BulkOperation.Builder().update(
7
i -> i.action(new UpdateAction.Builder<>().doc(doc).docAsUpsert(true).build()).id("1"))
8
.build();
9
10
List<BulkOperation> list = Collections.singletonList(op);
11
BulkResponse response = esClient.bulk(bulkBuilder -> bulkBuilder.index("products").operations(list));
参考文档:
1、Elasticsearch 官方文档:
2、Github文档