com.stratio.ingestion.serializer.elasticsearch.ElasticSearchSerializerWithMappingTest.java Source code

Java tutorial

Introduction

Here is the source code for com.stratio.ingestion.serializer.elasticsearch.ElasticSearchSerializerWithMappingTest.java

Source

/**
 * Copyright (C) 2014 Stratio (http://stratio.com)
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *         http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.stratio.ingestion.serializer.elasticsearch;

import static org.apache.flume.sink.elasticsearch.ElasticSearchEventSerializer.charset;

import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.util.List;
import java.util.Map;

import org.apache.commons.io.FileUtils;
import org.apache.commons.io.IOUtils;
import org.apache.flume.Context;
import org.apache.flume.Event;
import org.apache.flume.event.EventBuilder;
import org.apache.flume.sink.elasticsearch.ElasticSearchIndexRequestBuilderFactory;
import org.elasticsearch.action.admin.cluster.state.ClusterStateResponse;
import org.elasticsearch.client.Client;
import org.elasticsearch.cluster.metadata.IndexMetaData;
import org.elasticsearch.cluster.metadata.MappingMetaData;
import org.elasticsearch.common.collect.ImmutableOpenMap;
import org.elasticsearch.common.collect.Maps;
import org.elasticsearch.node.Node;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.typesafe.config.Config;
import com.typesafe.config.ConfigFactory;

public class ElasticSearchSerializerWithMappingTest {

    private static final Logger LOGGER = LoggerFactory.getLogger(ElasticSearchSerializerWithMappingTest.class);
    private final static String INDEX_PREFIX = "flume";
    private final static String INDEX_TYPE = "log";
    private final static String MAPPING_PATH = "/mapping.json";
    private final static String BAD_MAPPING_PATH = "/mapping2.json";
    private final static String TIMESTAMP_HEADER = "timestamp";
    private final static long DAY_IN_MILLIS = 86400000;

    protected static Config conf;
    private Node node;
    private ElasticSearchSerializerWithMapping serializer;
    private Client client;
    private Client elasticSearchClient;
    private String expectedESMapping = "";

    @Before
    public void setUpES() throws IOException {
        FileUtils.deleteDirectory(new File("data/test"));
        String jsonMapping = IOUtils.toString(this.getClass().getResourceAsStream(MAPPING_PATH));
        expectedESMapping = trimAllWhitespace("{\"" + INDEX_TYPE + "\":" + jsonMapping + "}");

        conf = ConfigFactory.load();

        //LOGGER.debug("Connecting to Elastic Search: " + conf.getStringList("elasticsearch.hosts").toString());

        List<String> elasticSearchHosts = conf.getStringList("elasticsearch.hosts");
        String elasticSearchClusterName = conf.getString("elasticsearch.clusterName");

    }

    @Test(expected = NullPointerException.class)
    public void badFileTest() throws IOException {

        URL resourceUrl = getClass().getResource(MAPPING_PATH);
        Context context = new Context();
        context.put("mappingFile2", resourceUrl.getPath());
        serializer = new ElasticSearchSerializerWithMapping();
        serializer.configure(context);
    }

    @Test
    public void goodFileTest() throws IOException {
        URL resourceUrl = getClass().getResource(MAPPING_PATH);
        Context context = new Context();
        context.put("mappingFile", resourceUrl.getPath());
        serializer = new ElasticSearchSerializerWithMapping();
        serializer.configure(context);

        Assert.assertNotNull(serializer);
    }

    @Test(expected = NullPointerException.class)
    public void sameTimestampEventsShouldCreateOnlyOneIndexWithTheExpectedMapping() throws IOException {
        Event event = createExampleEvent(System.currentTimeMillis());
        String indexName = getIndexName(INDEX_PREFIX, new TimeStampedEvent(event).getTimestamp());

        serializer.createIndexRequest(elasticSearchClient, INDEX_PREFIX, INDEX_TYPE, event);
        serializer.createIndexRequest(elasticSearchClient, INDEX_PREFIX, INDEX_TYPE, event);
        ImmutableOpenMap<String, MappingMetaData> mappings = getIndices().get(indexName).getMappings();
        String mappingActual = mappings.get(INDEX_TYPE).source().string();

        Assert.assertTrue("The index must exists and its mapping must be the same as the expected",
                mappingActual.equals(expectedESMapping));
    }

    @Test(expected = NullPointerException.class)
    public void differentTimestampEventsShouldCreateDifferentIndecesWithTheExpectedMapping() throws IOException {
        long timestamp = System.currentTimeMillis();
        Event event1 = createExampleEvent(timestamp);
        String indexName1 = getIndexName(INDEX_PREFIX, new TimeStampedEvent(event1).getTimestamp());
        Event event2 = createExampleEvent(timestamp + DAY_IN_MILLIS);
        String indexName2 = getIndexName(INDEX_PREFIX, new TimeStampedEvent(event2).getTimestamp());

        serializer.createIndexRequest(elasticSearchClient, INDEX_PREFIX, INDEX_TYPE, event1);
        serializer.createIndexRequest(elasticSearchClient, INDEX_PREFIX, INDEX_TYPE, event2);
        ImmutableOpenMap<String, IndexMetaData> indices = getIndices();
        ImmutableOpenMap<String, MappingMetaData> mappingsIndex1 = indices.get(indexName1).getMappings();
        ImmutableOpenMap<String, MappingMetaData> mappingsIndex2 = indices.get(indexName2).getMappings();
        String mappingIndex1 = mappingsIndex1.get(INDEX_TYPE).source().string();
        String mappingIndex2 = mappingsIndex2.get(INDEX_TYPE).source().string();

        Assert.assertTrue("The first index must exists and its mapping must be the same as the expected",
                mappingIndex1.equals(expectedESMapping));
        Assert.assertTrue("The second index must exists and its mapping must be the same as the expected",
                mappingIndex2.equals(expectedESMapping));
    }

    private String getIndexName(String indexPrefix, long timestamp) {
        return new StringBuilder(indexPrefix).append('-')
                .append(ElasticSearchIndexRequestBuilderFactory.df.format(timestamp)).toString();
    }

    private Event createExampleEvent(long timestamp) {
        String message = "test body";
        Map<String, String> headers = Maps.newHashMap();
        headers.put(TIMESTAMP_HEADER, String.valueOf(timestamp));
        Event event = EventBuilder.withBody(message.getBytes(charset));
        event.setHeaders(headers);
        return event;
    }

    private ImmutableOpenMap<String, IndexMetaData> getIndices() {
        ClusterStateResponse clusterStateResponse = elasticSearchClient.admin().cluster().prepareState().execute()
                .actionGet();
        return clusterStateResponse.getState().getMetaData().getIndices();
    }

    private String trimAllWhitespace(String str) {
        if (!hasLength(str)) {
            return str;
        }
        StringBuilder sb = new StringBuilder(str);
        int index = 0;
        while (sb.length() > index) {
            if (Character.isWhitespace(sb.charAt(index))) {
                sb.deleteCharAt(index);
            } else {
                index++;
            }
        }
        return sb.toString();
    }

    private boolean hasLength(CharSequence str) {
        return (str != null && str.length() > 0);
    }

}