DLI outputs the output data of the Flink job to an index in the Elasticsearch engine of the Cloud Search Service (CSS).
Elasticsearch is a popular enterprise-class Lucene-powered search server and provides the distributed multi-user capabilities. It delivers multiple functions, including full-text retrieval, structured search, analytics, aggregation, and highlighting. With Elasticsearch, you can achieve stable, reliable, real-time search. Elasticsearch applies to diversified scenarios, such as log analysis and site search.
CSS is a fully managed, distributed search service. It is fully compatible with open-source Elasticsearch and provides DLI with structured and unstructured data search, statistics, and report capabilities.
For details, see Elasticsearch SQL Connector.
Type |
Description |
|---|---|
Supported Table Types |
Result table |
Supported Data Formats |
create table esSink (
attr_name attr_type
(',' attr_name attr_type)*
(','PRIMARY KEY (attr_name, ...) NOT ENFORCED)
)
with (
'connector' = 'elasticsearch-7',
'hosts' = '',
'index' = ''
);
Parameter |
Mandatory |
Default Value |
Data Type |
Description |
|---|---|---|---|---|
connector |
Yes |
None |
String |
Specify what connector to use. Set this parameter to elasticsearch-7. This indicates connecting to Elasticsearch 7.x cluster. |
hosts |
Yes |
None |
String |
Host name of the cluster where Elasticsearch is located. Use semicolons (;) to separate multiple host names. |
index |
Yes |
None |
String |
Elasticsearch index for every record. The index can be a static index (for example, 'myIndex') or a dynamic index (for example, 'index-{log_ts|yyyy-MM-dd}'). See the following Dynamic Index section for more details. |
username |
No |
None |
String |
Username of the cluster where Elasticsearch locates. This parameter must be configured in pair with password. |
password |
No |
None |
String |
Password of the cluster where Elasticsearch locates. This parameter must be configured in pair with username. |
document-id.key-delimiter |
No |
_ |
String |
Delimiter for composite keys ("_" by default), e.g., $ would result in IDs KEY1$KEY2$KEY3. |
failure-handler |
No |
fail |
String |
Failure handling strategy in case a request to Elasticsearch fails. Valid strategies are:
|
sink.flush-on-checkpoint |
No |
true |
Boolean |
Flush on checkpoint or not. When disabled, a sink will not wait for all pending action requests to be acknowledged by Elasticsearch on checkpoints. Thus, a sink does not provide any strong guarantees for at-least-once delivery of action requests. |
sink.bulk-flush.max-actions |
No |
1000 |
Interger |
Maximum number of buffered actions per bulk request. You can set this parameter to 0 to disable it. |
sink.bulk-flush.max-size |
No |
2mb |
MemorySize |
Maximum size in memory of buffered actions per bulk request. Must be in MB granularity. Can be set to 0 to disable it. |
sink.bulk-flush.interval |
No |
1s |
Duration |
The interval to flush buffered actions. Can be set to 0 to disable it. Note, both sink.bulk-flush.max-size and sink.bulk-flush.max-actions can be set to 0 with the flush interval set allowing for complete async processing of buffered actions. |
sink.bulk-flush.backoff.strategy |
No |
DISABLED |
String |
Specify how to perform retries if any flush actions failed due to a temporary request error. Valid strategies are:
|
sink.bulk-flush.backoff.max-retries |
No |
None |
Integer |
Maximum number of rollback retries. |
sink.bulk-flush.backoff.delay |
No |
None |
Duration |
Delay between each backoff attempt. For CONSTANT backoff, this is simply the delay between each retry. For EXPONENTIAL backoff, this is the initial base delay. |
connection.path-prefix |
No |
None |
String |
Prefix string added to each REST communication, for example, '/v1'. |
connection.request-timeout |
No |
None |
Duration |
The timeout in milliseconds for requesting a connection from the connection manager. The timeout must be larger than or equal to 0. A timeout value of zero is interpreted as an infinite timeout. |
connection.timeout |
No |
None |
Duration |
The timeout in milliseconds for establishing a connection. The timeout must be larger than or equal to 0. A timeout value of zero is interpreted as an infinite timeout. |
socket.timeout |
No |
None |
Duration |
The socket timeout (SO_TIMEOUT) for waiting for data. The timeout must be larger than or equal to 0. A timeout value of zero is interpreted as an infinite timeout. |
format |
No |
json |
String |
Elasticsearch connector supports to specify a format. The format must produce a valid JSON document. By default, the built-in JSON format is used. Refer to Format for more details and format parameters. |
certificate |
No |
None |
String |
Location of the Elasticsearch cluster certificate in OBS. This parameter is required only when the security mode and HTTPS are enabled. Download the certificate from the CSS management console and upload the certificate to OBS. This parameter specifies the OBS address. Example: obs://bucket/path/CloudSearchService.cer |
The Elasticsearch sink can work in either upsert mode or append mode, depending on whether a primary key is defined.
In the Elasticsearch connector, the primary key is used to calculate the Elasticsearch document ID, which is a string of up to 512 bytes. It cannot have whitespaces.
The Elasticsearch connector generates a document ID string for every row by concatenating all primary key fields in the order defined in the DDL using a key delimiter specified by document-id.key-delimiter. Certain types are not allowed as a primary key field as they do not have a good string representation, e.g. BYTES, ROW, ARRAY, MAP, etc.
If no primary key is specified, Elasticsearch will generate a document ID automatically.
The Elasticsearch sink supports both static index and dynamic index.
When using the dynamic index generated by the current system time, for changelog stream, there is no guarantee that the records with the same primary key can generate the same index name. Therefore, the dynamic index based on the system time can only support append only stream.
In this example, data is read from the Kafka data source and written to the Elasticsearch result table (Elasticsearch 7.10.2). The procedure is as follows:
PUT /orders
{
"settings": {
"number_of_shards": 1
},
"mappings": {
"properties": {
"order_id": {
"type": "text"
},
"order_channel": {
"type": "text"
},
"order_time": {
"type": "text"
},
"pay_amount": {
"type": "double"
},
"real_pay": {
"type": "double"
},
"pay_time": {
"type": "text"
},
"user_id": {
"type": "text"
},
"user_name": {
"type": "text"
},
"area_id": {
"type": "text"
}
}
}
}
CREATE TABLE kafkaSource ( order_id string, order_channel string, order_time string, pay_amount double, real_pay double, pay_time string, user_id string, user_name string, area_id string ) WITH ( 'connector' = 'kafka', 'topic' = 'KafkaTopic', 'properties.bootstrap.servers' = 'KafkaAddress1:KafkaPort,KafkaAddress2:KafkaPort', 'properties.group.id' = 'GroupId', 'scan.startup.mode' = 'latest-offset', 'format' = 'json' ); CREATE TABLE elasticsearchSink ( order_id string, order_channel string, order_time string, pay_amount double, real_pay double, pay_time string, user_id string, user_name string, area_id string ) WITH ( 'connector' = 'elasticsearch-7', 'hosts' = 'ElasticsearchAddress:ElasticsearchPort', 'index' = 'orders' ); insert into elasticsearchSink select * from kafkaSource;
{"order_id":"202103241000000001", "order_channel":"webShop", "order_time":"2021-03-24 10:00:00", "pay_amount":"100.00", "real_pay":"100.00", "pay_time":"2021-03-24 10:02:03", "user_id":"0001", "user_name":"Alice", "area_id":"330106"}
{"order_id":"202103241606060001", "order_channel":"appShop", "order_time":"2021-03-24 16:06:06", "pay_amount":"200.00", "real_pay":"180.00", "pay_time":"2021-03-24 16:10:06", "user_id":"0001", "user_name":"Alice", "area_id":"330106"}
GET orders/_search
{
"took" : 201,
"timed_out" : false,
"_shards" : {
"total" : 1,
"successful" : 1,
"skipped" : 0,
"failed" : 0
},
"hits" : {
"total" : {
"value" : 2,
"relation" : "eq"
},
"max_score" : 1.0,
"hits" : [
{
"_index" : "orders",
"_type" : "_doc",
"_id" : "fopyx4sBUuT2wThgYGcp",
"_score" : 1.0,
"_source" : {
"order_id" : "202103241606060001",
"order_channel" : "appShop",
"order_time" : "2021-03-24 16:06:06",
"pay_amount" : 200.0,
"real_pay" : 180.0,
"pay_time" : "2021-03-24 16:10:06",
"user_id" : "0001",
"user_name" : "Alice",
"area_id" : "330106"
}
},
{
"_index" : "orders",
"_type" : "_doc",
"_id" : "f4pyx4sBUuT2wThgYGcr",
"_score" : 1.0,
"_source" : {
"order_id" : "202103241000000001",
"order_channel" : "webShop",
"order_time" : "2021-03-24 10:00:00",
"pay_amount" : 100.0,
"real_pay" : 100.0,
"pay_time" : "2021-03-24 10:02:03",
"user_id" : "0001",
"user_name" : "Alice",
"area_id" : "330106"
}
}
]
}
}