Java tutorial
/** * Copyright 2013 Twitter, Inc. * 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.rsinghal.cep.sample.twitter; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.util.ArrayList; import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.Properties; import java.util.concurrent.BlockingQueue; import java.util.concurrent.ConcurrentSkipListMap; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.LinkedBlockingQueue; import java.util.concurrent.atomic.AtomicLong; import kafka.javaapi.producer.Producer; import kafka.message.Message; import kafka.producer.KeyedMessage; import kafka.producer.ProducerConfig; import redis.clients.jedis.Jedis; import org.apache.avro.Schema; import org.apache.avro.io.DatumWriter; import org.apache.avro.io.Encoder; import org.apache.avro.io.EncoderFactory; import org.apache.avro.specific.SpecificDatumWriter; import org.apache.commons.lang.ArrayUtils; import org.apache.commons.lang.StringUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Required; import org.springframework.jmx.export.annotation.ManagedAttribute; import org.springframework.jmx.export.annotation.ManagedResource; import org.springframework.scheduling.annotation.Scheduled; import twitter4j.FilterQuery; import twitter4j.StallWarning; import twitter4j.Status; import twitter4j.StatusDeletionNotice; import twitter4j.StatusListener; import twitter4j.StatusUpdate; import twitter4j.Twitter; import twitter4j.TwitterException; import twitter4j.TwitterFactory; import twitter4j.TwitterObjectFactory; import twitter4j.TwitterStream; import twitter4j.TwitterStreamFactory; import twitter4j.UserMentionEntity; import twitter4j.auth.AccessToken; import twitter4j.conf.ConfigurationBuilder; import twitter4j.json.DataObjectFactory; import com.google.common.collect.Lists; import com.rsinghal.cep.sample.twitter.avro.v11.TwitterStatusUpdate; import com.rsinghal.cep.sample.twitter.v11.TwitterStatusUpdateConverter; import com.twitter.hbc.ClientBuilder; import com.twitter.hbc.core.Constants; import com.twitter.hbc.core.endpoint.DefaultStreamingEndpoint; import com.twitter.hbc.core.endpoint.StatusesFilterEndpoint; import com.twitter.hbc.core.processor.StringDelimitedProcessor; import com.twitter.hbc.httpclient.BasicClient; import com.twitter.hbc.httpclient.auth.Authentication; import com.twitter.hbc.httpclient.auth.OAuth1; import com.twitter.hbc.twitter4j.Twitter4jStatusClient; import com.twitter.hbc.twitter4j.handler.StatusStreamHandler; import com.twitter.hbc.twitter4j.message.DisconnectMessage; import com.twitter.hbc.twitter4j.message.StallWarningMessage; @ManagedResource(description = "@mybankrbs") public class TwitterBot { private final static Logger logger = LoggerFactory.getLogger(StreamCollector.class); private BlockingQueue<String> queue; private BasicClient client; private Twitter twitter = null; private AtomicLong tweetCounterLast = new AtomicLong(0); private AtomicLong tweetCounter = new AtomicLong(0); private AtomicLong deletionNoticeCounter = new AtomicLong(0); private AtomicLong trackLimitationNoticeCounter = new AtomicLong(0); private AtomicLong scrubGeoCounter = new AtomicLong(0); private AtomicLong exceptionCounter = new AtomicLong(0); private AtomicLong disconnectMessageCounter = new AtomicLong(0); private AtomicLong unknownMessageTypeCounter = new AtomicLong(0); private AtomicLong stallWarningCounter = new AtomicLong(0); private AtomicLong stallWarningMessageCounter = new AtomicLong(0); private String kafkaHostname; private String kafkaPort; private String kafkaTopicName; private int numberOfProcessingThreads = 5; private String consumerKey; private String consumerSecret; private String accessToken; private String accessTokenSecret; private String redisHostname; public String getRedisHostname() { return redisHostname; } public void setRedisHostname(String redisHostname) { this.redisHostname = redisHostname; } public int getRedisPort() { return redisPort; } public void setRedisPort(int redisPort) { this.redisPort = redisPort; } private int redisPort; private Jedis jedis; private List<String> trackTerms = new ArrayList<String>(); Schema schema11 = null; private final StatusesFilterEndpoint endpoint = new StatusesFilterEndpoint(); private final ConcurrentSkipListMap<Long, Long> tweetsAndReplies = new ConcurrentSkipListMap<>(); private final ConcurrentSkipListMap<String, Long> handleAndIds = new ConcurrentSkipListMap<>(); private final TwitterStream twitterStream = new TwitterStreamFactory().getInstance(); @ManagedAttribute(description = "Number of tweets processed since start", currencyTimeLimit = 15) public Long getTweetCounter() { return tweetCounter.longValue(); } @ManagedAttribute(description = "Number of deletion notice since start", currencyTimeLimit = 15) public Long getDeletionNoticeCounter() { return deletionNoticeCounter.longValue(); } @ManagedAttribute(description = "Number of track limitation notice since start", currencyTimeLimit = 15) public Long getTrackLimitiationNoticeCounter() { return trackLimitationNoticeCounter.longValue(); } @ManagedAttribute(description = "Number of scrub geo since start", currencyTimeLimit = 15) public Long getScrubGeoCounter() { return scrubGeoCounter.longValue(); } @ManagedAttribute(description = "Number of exception since start", currencyTimeLimit = 15) public Long getExceptionCounter() { return exceptionCounter.longValue(); } @ManagedAttribute(description = "Number of disconnect message since start", currencyTimeLimit = 15) public Long getDisconnectMessageCounter() { return disconnectMessageCounter.longValue(); } @ManagedAttribute(description = "Number of unknown message type since start", currencyTimeLimit = 15) public Long getUnknownMessageTypeCounter() { return unknownMessageTypeCounter.longValue(); } @ManagedAttribute(description = "Number of stall warning since start", currencyTimeLimit = 15) public Long getStallWarningCounter() { return stallWarningCounter.longValue(); } @ManagedAttribute(description = "Number of stall warning messsage since start", currencyTimeLimit = 15) public Long getStallWarningMessageCounter() { return stallWarningMessageCounter.longValue(); } @Required public void setKafkaTopicName(String kafkaTopicName) { this.kafkaTopicName = kafkaTopicName; } @Required public void setKafkaHostname(String kafkaHostname) { this.kafkaHostname = kafkaHostname; } @Required public void setKafkaPort(String kafkaPort) { this.kafkaPort = kafkaPort; } public void setNumberOfProcessingThreads(int numberOfProcessingThreads) { this.numberOfProcessingThreads = numberOfProcessingThreads; } @Required public void setConsumerKey(String consumerKey) { this.consumerKey = consumerKey; } @Required public void setConsumerSecret(String consumerSecret) { this.consumerSecret = consumerSecret; } @Required public void setAccessToken(String accessToken) { this.accessToken = accessToken; } @Required public void setAccessTokenSecret(String accessTokenSecret) { this.accessTokenSecret = accessTokenSecret; } public void setTrackTerms(List<String> trackTerms) { this.trackTerms = trackTerms; } private long[] userIdsToTrack; protected DefaultStreamingEndpoint createEndpoint() { endpoint.trackTerms(trackTerms); return endpoint; } /** * Write the counters to the log every 5min */ @Scheduled(fixedRate = 300000) public void logCounter() { logger.info( "-----------------------------------------------------------------------------------------------------------------"); logger.info("[Tweet=" + getTweetCounter() + "(+" + (getTweetCounter() - tweetCounterLast.get()) + ")" + ",Exc=" + getExceptionCounter() + ",Del=" + getDeletionNoticeCounter() + ",Disc=" + getDisconnectMessageCounter() + ",Scub=" + getScrubGeoCounter() + ",Stall=" + getStallWarningCounter() + ",Limit=" + getTrackLimitiationNoticeCounter() + ",Unknown=" + getUnknownMessageTypeCounter() + "]"); tweetCounterLast.set(tweetCounter.get()); } public TwitterBot() { Schema.Parser parser = new Schema.Parser(); try { schema11 = parser.parse(getClass().getResourceAsStream("/TwitterSchema-v1.1.avsc")); } catch (IOException e1) { // TODO Auto-generated catch block e1.printStackTrace(); } } public void start() { ConfigurationBuilder cb = new ConfigurationBuilder(); cb.setJSONStoreEnabled(true); twitter = new TwitterFactory(cb.build()).getInstance(); twitter.setOAuthConsumer(consumerKey, consumerSecret); twitter.setOAuthAccessToken(new AccessToken(accessToken, accessTokenSecret)); userIdsToTrack = new long[trackTerms.size()]; for (int i = 0; i < trackTerms.size(); i++) { try { String screenName = StringUtils.substringAfter(trackTerms.get(i), "@"); logger.info("Fetching user ID for - " + screenName); userIdsToTrack[i] = twitter.showUser(screenName).getId(); } catch (TwitterException e) { // TODO Auto-generated catch block e.printStackTrace(); } } // twitter = null; twitterStream.setOAuthConsumer(consumerKey, consumerSecret); twitterStream.setOAuthAccessToken(new AccessToken(accessToken, accessTokenSecret)); twitterStream.addListener(listener2); // sample() method internally creates a thread which manipulates TwitterStream and calls these adequate listener methods continuously. twitterStream.filter(new FilterQuery(0, userIdsToTrack, trackTerms.toArray(new String[trackTerms.size()]))); jedis = new Jedis(redisHostname, redisPort, 1800); jedis.connect(); System.out.println("start() ..."); /* // Create an appropriately sized blocking queue BlockingQueue<String> queue = new LinkedBlockingQueue<String>(10000); // create the endpoint DefaultStreamingEndpoint endpoint = createEndpoint(); System.out.println("endpoint created ..."); endpoint.stallWarnings(false); // create an authentication Authentication auth = new OAuth1(consumerKey, consumerSecret, accessToken, accessTokenSecret); // Create a new BasicClient. By default gzip is enabled. client = new ClientBuilder().name("sampleExampleClient") .hosts(Constants.STREAM_HOST).endpoint(endpoint) .authentication(auth) .processor(new StringDelimitedProcessor(queue)).build(); System.out.println("client created ..."); // Create an executor service which will spawn threads to do the actual // work of parsing the incoming messages and // calling the listeners on each message ExecutorService service = Executors .newFixedThreadPool(this.numberOfProcessingThreads); // Wrap our BasicClient with the twitter4j client Twitter4jStatusClient t4jClient = new Twitter4jStatusClient(client, queue, Lists.newArrayList(listener2), service); // Establish a connection t4jClient.connect(); System.out.println("connection established ..."); for (int threads = 0; threads < this.numberOfProcessingThreads; threads++) { // This must be called once per processing thread t4jClient.process(); System.out.println("thread " + threads + " started ..."); }*/ } public void stop() { //client.stop(); twitterStream.shutdown(); } public boolean isRunning() { return true; } // A bare bones StatusStreamHandler, which extends listener and gives some // extra functionality private StatusListener listener2 = new StatusStreamHandler() { Producer<String, byte[]> producer = null; public void store(Status status) throws IOException, InterruptedException { final String zkConnection = kafkaHostname + ":" + kafkaPort; final String topic = kafkaTopicName; TwitterStatusUpdate update = TwitterStatusUpdateConverter.convert(status); ByteArrayOutputStream out = new ByteArrayOutputStream(); DatumWriter<TwitterStatusUpdate> writer = new SpecificDatumWriter<TwitterStatusUpdate>( TwitterStatusUpdate.class); Encoder encoder = EncoderFactory.get().jsonEncoder(schema11, out, true); writer.write(update, encoder); encoder.flush(); out.close(); Message message = new Message(out.toByteArray()); if (producer == null) { System.out.println("Connecting to Kafka Server: " + zkConnection + " using topic " + topic); logger.info("Connecting to Kafka Server: " + zkConnection + " using topic " + topic); Properties props = new Properties(); props.put("metadata.broker.list", zkConnection); props.put("request.required.acks", "1"); //props.put("serializer.class", "kafka.serializer.StringEncoder"); props.put("producer.type", "sync"); props.put("compression.codec", "1"); producer = new kafka.javaapi.producer.Producer<String, byte[]>(new ProducerConfig(props)); System.out.println( "Connected Sucessfully to Kafka Server: " + zkConnection + " using topic " + topic); logger.info("Connected Successfully to Kafka Server: " + zkConnection + " using topic " + topic); } producer.send(new KeyedMessage<String, byte[]>(topic, out.toByteArray())); } int i = 0; @Override public void onStatus(Status status) { boolean processIt = true; //String retweetedUser = ""; if (status == null) { System.err.println("status is null"); } if (processIt) { try { //store(status); long replyUserId = status.getInReplyToUserId(); long userId = status.getUser().getId(); if (ArrayUtils.contains(userIdsToTrack, replyUserId) && !ArrayUtils.contains(userIdsToTrack, userId)) { Map<String, String> conversation = jedis.hgetAll(String.valueOf(userId)); if (conversation == null || conversation.isEmpty()) { initiateConversation(status); } else { classifyAndRespond(status, conversation); } } /*//twitterStream.filter(new F); long tweetId = status.getId(); long replyTweetId = status.getInReplyToStatusId(); tweetsAndReplies.put(tweetId, replyTweetId); for(UserMentionEntity e : status.getUserMentionEntities()) { if(trackTerms.contains("@"+e.getScreenName())) { if(handleAndIds.size()<trackTerms.size()) { handleAndIds.put("@"+e.getScreenName(), e.getId()); } } } TwitterStream twitterStream = new TwitterStreamFactory().getInstance(); twitterStream.addListener(listener2); //twitterStream. */ // increment the counter of tweets tweetCounter.incrementAndGet(); //} catch (IOException e) { // TODO Auto-generated catch block //logger.error("Error occured in onStatus()", e); //e.printStackTrace(); //} catch (InterruptedException e) { //logger.error("Error occured in onStatus()", e); // TODO Auto-generated catch block //e.printStackTrace(); } catch (NullPointerException e) { logger.error("Error occured in onStatus()", e); e.printStackTrace(); } } } private void initiateConversation(Status status) { putInRedis(status, 1); try { //reply to that tweet StatusUpdate statusUpdate = new StatusUpdate("Hi @" + status.getUser().getScreenName() + ". Did I get your correct name Ms " + status.getUser().getName() + " ?"); statusUpdate.inReplyToStatusId(status.getId()); putInRedis(twitter.updateStatus(statusUpdate), 2); } catch (Exception e) { e.printStackTrace(); } } private void putInRedis(Status status, int index) { HashMap<String, String> statusData = new HashMap<String, String>() { { put(String.valueOf(status.getId()), TwitterObjectFactory.getRawJSON(status)); put(String.valueOf(index), String.valueOf(status.getId())); } }; jedis.hmset(String.valueOf(status.getUser().getId()), statusData); } private void classifyAndRespond(Status status, Map<String, String> conversation) { } @Override public void onDeletionNotice(StatusDeletionNotice statusDeletionNotice) { //too many, do not log //logger.info("=====> onDeletionNotice: " + statusDeletionNotice); deletionNoticeCounter.incrementAndGet(); } @Override public void onTrackLimitationNotice(int limit) { logger.warn("=====> onTrackLimitationNotice: " + limit); trackLimitationNoticeCounter.incrementAndGet(); } @Override public void onScrubGeo(long user, long upToStatus) { logger.warn("=====> onScrubGeo: " + user); scrubGeoCounter.incrementAndGet(); } @Override public void onException(Exception e) { logger.error("=====> onException", e); exceptionCounter.incrementAndGet(); } @Override public void onDisconnectMessage(DisconnectMessage message) { logger.error("=====> onDisconnectMessage: " + message); disconnectMessageCounter.incrementAndGet(); } @Override public void onUnknownMessageType(String s) { logger.warn("=====> onUnknownMessageType: " + s); unknownMessageTypeCounter.incrementAndGet(); } @Override public void onStallWarning(StallWarning arg0) { logger.warn("=====> onStallWarning: " + arg0); stallWarningCounter.incrementAndGet(); } @Override public void onStallWarningMessage(StallWarningMessage arg0) { logger.warn("=====> onStallWarningMessage: " + arg0); stallWarningMessageCounter.incrementAndGet(); } }; }