All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.treasure_data.logger.sender.HttpSender Maven / Gradle / Ivy

The newest version!
//
// Treasure Data Logger for Java.
//
// Copyright (C) 2011 - 2013 Muga Nishizawa
//
//    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.treasure_data.logger.sender;

import java.io.IOException;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.regex.Pattern;

import org.fluentd.logger.sender.Sender;
import org.msgpack.MessagePack;

import com.treasure_data.client.TreasureDataClient;

public class HttpSender implements Sender {
    private static Logger LOG = Logger.getLogger(HttpSender.class.getName());

    private static Pattern databasePat = Pattern.compile("^([a-z0-9_]+)$");

    private static Pattern columnPat = Pattern.compile("^([a-z0-9_]+)$");

    public static boolean validateDatabaseName(String name) {
        if (name == null || name.equals("")) {
            LOG.info(String.format("Empty name is not allowed: %s",
                    new Object[] { name }));
            return false;
        }
        int len = name.length();
        if (len < 3 || 32 < len) {
            LOG.info(String.format("Name must be 3 to 32 characters, got %d characters.",
                    new Object[] { len }));
            return false;
        }

        if (!databasePat.matcher(name).matches()) {
            LOG.info("Name must consist only of alphabets, numbers, '_'.");
            return false;
        }

        return true;
    }

    public static boolean validateTableName(String name) {
        return validateDatabaseName(name);
    }

    public static boolean validateColumnName(String name) {
        if (name == null || name.equals("")) {
            LOG.info(String.format("Empty column name is not allowed: %s",
                    new Object[] { name }));
            return false;
        }

        int len = name.length();
        if (32 < len) {
            LOG.info(String.format("Column name must be to 32 characters, got %d characters.",
                    new Object[] { len }));
            return false;
        }

        if (!columnPat.matcher(name).matches()) {
            LOG.info("Column name must consist only of alphabets, numbers, '_'.");
            return false;
        }

        return true;
    }

    private MessagePack msgpack;

    private Map chunks;

    private int chunkLimit = 8 * 1024 * 1024; // 8MB

    protected Properties props;
    protected String host;
    protected int port;
    protected String apiKey;

    LinkedBlockingQueue queue;
    private int queueLimit = 50;
    private HttpSenderThread senderThread;

    private String name;

    public HttpSender(Properties props, final String host, final int port, final String apiKey) {
        this.props = props;
        if (apiKey == null) {
            throw new IllegalArgumentException("APIKey is required");
        }

        msgpack = new MessagePack();
        chunks = new ConcurrentHashMap();
        this.host = host;
        this.port = port;
        this.apiKey = apiKey;
        name = String.format("%s_%d", host, port);
    }

    public void startBackgroundProcess() {
        queue = new LinkedBlockingQueue();
        TreasureDataClient client = new TreasureDataClient(props);
        senderThread = new HttpSenderThread(this, client);
        new Thread(senderThread).start();
    }

    public boolean emit(String tag, Map record) {
        if (record.containsKey("time")) {
            return emit0(tag, record);
        } else {
            return emit(tag, System.currentTimeMillis() / 1000, record);
        }
    }

    public boolean emit(String tag, long timestamp, Map record) {
        record.put("time", timestamp);
        return emit0(tag, record);
    }

    private boolean emit0(String tag, Map record) {
        if (LOG.isLoggable(Level.FINE)) {
            LOG.fine(String.format("Emit event{tag=%s,record=%s}",
                    new Object[] { tag, record.toString() }));
        }

        String[] splited = tag.split("\\.");
        String databaseName = splited[splited.length - 2];
        String tableName = splited[splited.length - 1];

        // validation
        if (!validateDatabaseName(databaseName)) {
            String msg = String.format("Invalid database name %s", new Object[] { databaseName });
            LOG.severe(msg);
            throw new IllegalArgumentException(msg);
        }
        if (!validateTableName(tableName)) {
            String msg = String.format("Invalid table name %s", new Object[] { tableName });
            LOG.severe(msg);
            throw new IllegalArgumentException(msg);
        }

        // check queue limit
        if (getQueueSize() > queueLimit) {
            LOG.severe("queue length exceeds limit. cannot add new event log");
            return false;
        }

        String key = databaseName + "." + tableName;
        ExtendedPacker packer = null;
        try {
            packer = getPacker(key);
        } catch (IOException e) {
            LOG.log(Level.SEVERE, "Cannot create packer object", e);
            return false;
        }

        // write data to chunk
        try {
            packer.write(record);
        } catch (Exception e) {
            LOG.log(Level.SEVERE, String.format("Cannot write record to the (%s.%s) packer object:", databaseName, tableName), e);
            LOG.log(Level.SEVERE, record.toString());

            // Since packer object is refreshed, the remove method is not necessary. but, just in case,
            // if some problem occurred, it removes packer object itself.
            chunks.remove(key);
            return false;
        }

        if (packer.getChunkSize() > chunkLimit) {
            putQueue(databaseName, tableName, packer);
            LOG.fine(String.format("Put event on queue (size: %d)", getQueueSize()));
        }

        return true;
    }

    protected int getQueueSize() {
        return queue.size();
    }

    protected void putQueue(String databaseName, String tableName, ExtendedPacker packer) {
        try {
            long rowCount = packer.getRowCount();
            byte[] bytes = packer.getByteArray(); // it should be called after getRowCount is called.
            queue.put(new QueueEvent(databaseName, tableName, bytes, rowCount));

        } catch (IOException e) { // ignore
            LOG.log(Level.WARNING, "Cannot execute getByteArray()", e);

        } catch (InterruptedException e) { // ignore
        }
    }

    private synchronized ExtendedPacker getPacker(String key) throws IOException {
        ExtendedPacker packer;
        if (!chunks.containsKey(key)) {
            packer = new ExtendedPacker(msgpack);
            chunks.put(key, packer);
        } else {
            packer = chunks.get(key);
        }
        return packer;
    }

    public byte[] getBuffer() { // TODO #MN need the impl. for testing
        throw new UnsupportedOperationException();
    }

    public void flush() {
        try {
            flush0(false);
        } catch (IOException e) {
            // ignore
        } finally {
            senderThread.flush();
        }
    }

    public void close() {
        try {
            flush0(false);
        } catch (IOException e) {
            // ignore
        } finally {
            senderThread.stop();
        }
    }

    synchronized void flush0(boolean isThread) throws IOException {
        if (!chunks.isEmpty()) {
            for (Map.Entry entry : chunks.entrySet()) {
                String[] splited = entry.getKey().split("\\.");
                String databaseName = splited[0];
                String tableName = splited[1];
                ExtendedPacker packer = entry.getValue();
                putQueue(databaseName, tableName, packer);
            }
        }
        if (!isThread) {
            senderThread.flush();
        }
    }

    @Override
    public String getName() {
        return name;
    }

    @Override
    public boolean isConnected() {
        return false; // is not used at all.
    }

    @Override
    public String toString() {
        return getName();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy