/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you 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 org.apache.flume.client.avro; import java.io.File; import java.io.FileInputStream; import java.io.FileReader; import java.io.IOException; import java.io.InputStreamReader; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Properties; import com.google.common.base.Preconditions; import org.apache.commons.cli.CommandLine; import org.apache.commons.cli.CommandLineParser; import org.apache.commons.cli.GnuParser; import org.apache.commons.cli.HelpFormatter; import org.apache.commons.cli.Options; import org.apache.commons.cli.ParseException; import org.apache.flume.Event; import org.apache.flume.EventDeliveryException; import org.apache.flume.FlumeException; import org.apache.flume.annotations.InterfaceAudience; import org.apache.flume.annotations.InterfaceStability; import org.apache.flume.api.RpcClient; import org.apache.flume.api.RpcClientFactory; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @InterfaceAudience.Private @InterfaceStability.Evolving public class AvroCLIClient { private static final Logger logger = LoggerFactory .getLogger(AvroCLIClient.class); private static final int BATCH_SIZE = 5; private static final int MAX_LINE_LENGTH = 2000; private String hostname; private int port; private String fileName; private String rpcClientPropsFile; private String dirName; private Map<String, String> headers = new HashMap<String, String>(); private int sent; public static void main(String[] args) { AvroCLIClient client = new AvroCLIClient(); try { if (client.parseCommandLine(args)) { client.run(); } } catch (ParseException e) { logger.error("Unable to parse command line options - {}", e.getMessage()); } catch (IOException e) { logger.error("Unable to send data to Flume. Exception follows.", e); } catch (FlumeException e) { logger.error("Unable to open connection to Flume. Exception follows.", e); } catch (EventDeliveryException e) { logger.error("Unable to deliver events to Flume. Exception follows.", e); } logger.debug("Exiting"); } /* * Header Format : key1=value1, key2=value2,... */ private void parseHeaders(CommandLine commandLine) { String headerFile = commandLine.getOptionValue("headerFile"); FileInputStream fs = null; try { if (headerFile != null) { fs = new FileInputStream(headerFile); Properties properties = new Properties(); properties.load(fs); for (Map.Entry<Object, Object> propertiesEntry : properties.entrySet()) { String key = (String) propertiesEntry.getKey(); String value = (String) propertiesEntry.getValue(); logger.debug("Inserting Header Key [" + key + "] header value [" + value + "]"); headers.put(key, value); } } } catch (Exception e) { logger.error("Unable to load headerFile", headerFile, e); return; } finally { if (fs != null) { try { fs.close(); } catch (Exception e) { logger.error("Unable to close headerFile", e); return; } } } } private boolean parseCommandLine(String[] args) throws ParseException { Options options = new Options(); options .addOption("P", "rpcProps", true, "RPC client properties file with " + "server connection params") .addOption("p", "port", true, "port of the avro source") .addOption("H", "host", true, "hostname of the avro source") .addOption("F", "filename", true, "file to stream to avro source") .addOption(null, "dirname", true, "directory to stream to avro source") .addOption("R", "headerFile", true, ("file containing headers as " + "key/value pairs on each new line")) .addOption("h", "help", false, "display help text"); CommandLineParser parser = new GnuParser(); CommandLine commandLine = parser.parse(options, args); if (commandLine.hasOption('h')) { new HelpFormatter().printHelp("flume-ng avro-client", "", options, "The --dirname option assumes that a spooling directory exists " + "where immutable log files are dropped.", true); return false; } if (commandLine.hasOption("filename") && commandLine.hasOption("dirname")) { throw new ParseException( "--filename and --dirname options cannot be used simultaneously"); } if (!commandLine.hasOption("port") && !commandLine.hasOption("host") && !commandLine.hasOption("rpcProps")) { throw new ParseException("Either --rpcProps or both --host and --port " + "must be specified."); } if (commandLine.hasOption("rpcProps")) { rpcClientPropsFile = commandLine.getOptionValue("rpcProps"); Preconditions.checkNotNull(rpcClientPropsFile, "RPC client properties " + "file must be specified after --rpcProps argument."); Preconditions.checkArgument(new File(rpcClientPropsFile).exists(), "RPC client properties file %s does not exist!", rpcClientPropsFile); } if (rpcClientPropsFile == null) { if (!commandLine.hasOption("port")) { throw new ParseException( "You must specify a port to connect to with --port"); } port = Integer.parseInt(commandLine.getOptionValue("port")); if (!commandLine.hasOption("host")) { throw new ParseException( "You must specify a hostname to connect to with --host"); } hostname = commandLine.getOptionValue("host"); } fileName = commandLine.getOptionValue("filename"); dirName = commandLine.getOptionValue("dirname"); if (commandLine.hasOption("headerFile")) { parseHeaders(commandLine); } return true; } private void run() throws IOException, FlumeException, EventDeliveryException { EventReader reader = null; RpcClient rpcClient; if (rpcClientPropsFile != null) { rpcClient = RpcClientFactory.getInstance(new File(rpcClientPropsFile)); } else { rpcClient = RpcClientFactory.getDefaultInstance(hostname, port, BATCH_SIZE); } try { if (fileName != null) { reader = new SimpleTextLineEventReader(new FileReader(new File(fileName))); } else if (dirName != null) { reader = new ReliableSpoolingFileEventReader.Builder() .spoolDirectory(new File(dirName)).build(); } else { reader = new SimpleTextLineEventReader(new InputStreamReader(System.in)); } long lastCheck = System.currentTimeMillis(); long sentBytes = 0; int batchSize = rpcClient.getBatchSize(); List<Event> events; while (!(events = reader.readEvents(batchSize)).isEmpty()) { for (Event event : events) { event.setHeaders(headers); sentBytes += event.getBody().length; sent++; long now = System.currentTimeMillis(); if (now >= lastCheck + 5000) { logger.debug("Packed {} bytes, {} events", sentBytes, sent); lastCheck = now; } } rpcClient.appendBatch(events); if (reader instanceof ReliableEventReader) { ((ReliableEventReader) reader).commit(); } } logger.debug("Finished"); } finally { if (reader != null) { logger.debug("Closing reader"); reader.close(); } logger.debug("Closing RPC client"); rpcClient.close(); } } }