require 'csv' # 'fastercsv' required on Ruby versions < 1.9 require 'fastercsv' unless Array.new.respond_to?(:to_csv) require 'htmlentities' require 't/collectable' require 't/printable' require 't/rcfile' require 't/requestable' require 'thor' module T class Search < Thor include T::Collectable include T::Printable include T::Requestable DEFAULT_NUM_RESULTS = 20 MAX_NUM_RESULTS = 200 MAX_SCREEN_NAME_SIZE = 20 MAX_USERS_PER_REQUEST = 20 check_unknown_options! def initialize(*) super @rcfile = RCFile.instance end desc "all QUERY", "Returns the #{DEFAULT_NUM_RESULTS} most recent Tweets that match the specified query." method_option "csv", :aliases => "-c", :type => :boolean, :default => false, :desc => "Output in CSV format." method_option "long", :aliases => "-l", :type => :boolean, :default => false, :desc => "Output in long format." method_option "number", :aliases => "-n", :type => :numeric, :default => DEFAULT_NUM_RESULTS def all(query) rpp = options['number'] || DEFAULT_NUM_RESULTS statuses = collect_with_rpp(rpp) do |opts| client.search(query, opts) end if options['csv'] say STATUS_HEADINGS.to_csv unless statuses.empty? statuses.each do |status| say [status.id, csv_formatted_time(status), status.from_user, HTMLEntities.new.decode(status.full_text)].to_csv end elsif options['long'] array = statuses.map do |status| [status.id, ls_formatted_time(status), "@#{status.from_user}", HTMLEntities.new.decode(status.full_text).gsub(/\n+/, ' ')] end format = options['format'] || STATUS_HEADINGS.size.times.map{"%s"} print_table_with_headings(array, STATUS_HEADINGS, format) else say unless statuses.empty? statuses.each do |status| print_status(status) end end end desc "favorites QUERY", "Returns Tweets you've favorited that match the specified query." method_option "csv", :aliases => "-c", :type => :boolean, :default => false, :desc => "Output in CSV format." method_option "long", :aliases => "-l", :type => :boolean, :default => false, :desc => "Output in long format." def favorites(query) opts = {:count => MAX_NUM_RESULTS} statuses = collect_with_max_id do |max_id| opts[:max_id] = max_id unless max_id.nil? client.favorites(opts) end statuses = statuses.select do |status| /#{query}/i.match(status.full_text) end print_statuses(statuses) end map %w(faves) => :favorites desc "list [USER/]LIST QUERY", "Returns Tweets on a list that match specified query." method_option "csv", :aliases => "-c", :type => :boolean, :default => false, :desc => "Output in CSV format." method_option "id", :aliases => "-i", :type => "boolean", :default => false, :desc => "Specify user via ID instead of screen name." method_option "long", :aliases => "-l", :type => :boolean, :default => false, :desc => "Output in long format." def list(list, query) owner, list = list.split('/') if list.nil? list = owner owner = @rcfile.active_profile[0] else owner = if options['id'] owner.to_i else owner.strip_ats end end opts = {:count => MAX_NUM_RESULTS} statuses = collect_with_max_id do |max_id| opts[:max_id] = max_id unless max_id.nil? client.list_timeline(owner, list, opts) end statuses = statuses.select do |status| /#{query}/i.match(status.full_text) end print_statuses(statuses) end desc "mentions QUERY", "Returns Tweets mentioning you that match the specified query." method_option "csv", :aliases => "-c", :type => :boolean, :default => false, :desc => "Output in CSV format." method_option "long", :aliases => "-l", :type => :boolean, :default => false, :desc => "Output in long format." def mentions(query) opts = {:count => MAX_NUM_RESULTS} statuses = collect_with_max_id do |max_id| opts[:max_id] = max_id unless max_id.nil? client.mentions(opts) end statuses = statuses.select do |status| /#{query}/i.match(status.full_text) end print_statuses(statuses) end map %w(replies) => :mentions desc "retweets QUERY", "Returns Tweets you've retweeted that match the specified query." method_option "csv", :aliases => "-c", :type => :boolean, :default => false, :desc => "Output in CSV format." method_option "long", :aliases => "-l", :type => :boolean, :default => false, :desc => "Output in long format." def retweets(query) opts = {:count => MAX_NUM_RESULTS} statuses = collect_with_max_id do |max_id| opts[:max_id] = max_id unless max_id.nil? client.retweeted_by(opts) end statuses = statuses.select do |status| /#{query}/i.match(status.full_text) end print_statuses(statuses) end map %w(rts) => :retweets desc "timeline [USER] QUERY", "Returns Tweets in your timeline that match the specified query." method_option "csv", :aliases => "-c", :type => :boolean, :default => false, :desc => "Output in CSV format." method_option "id", :aliases => "-i", :type => "boolean", :default => false, :desc => "Specify user via ID instead of screen name." method_option "long", :aliases => "-l", :type => :boolean, :default => false, :desc => "Output in long format." def timeline(*args) opts = {:count => MAX_NUM_RESULTS} query = args.pop user = args.pop if user user = if options['id'] user.to_i else user.strip_ats end statuses = collect_with_max_id do |max_id| opts[:max_id] = max_id unless max_id.nil? client.user_timeline(user, opts) end else statuses = collect_with_max_id do |max_id| opts[:max_id] = max_id unless max_id.nil? client.home_timeline(opts) end end statuses = statuses.select do |status| /#{query}/i.match(status.full_text) end print_statuses(statuses) end map %w(tl) => :timeline desc "users QUERY", "Returns users that match the specified query." method_option "csv", :aliases => "-c", :type => :boolean, :default => false, :desc => "Output in CSV format." method_option "favorites", :aliases => "-v", :type => :boolean, :default => false, :desc => "Sort by number of favorites." method_option "followers", :aliases => "-f", :type => :boolean, :default => false, :desc => "Sort by number of followers." method_option "friends", :aliases => "-e", :type => :boolean, :default => false, :desc => "Sort by number of friends." method_option "listed", :aliases => "-d", :type => :boolean, :default => false, :desc => "Sort by number of list memberships." method_option "long", :aliases => "-l", :type => :boolean, :default => false, :desc => "Output in long format." method_option "posted", :aliases => "-p", :type => :boolean, :default => false, :desc => "Sort by the time when Twitter account was posted." method_option "reverse", :aliases => "-r", :type => :boolean, :default => false, :desc => "Reverse the order of the sort." method_option "tweets", :aliases => "-t", :type => :boolean, :default => false, :desc => "Sort by number of Tweets." method_option "unsorted", :aliases => "-u", :type => :boolean, :default => false, :desc => "Output is not sorted." def users(query) users = 1.upto(50).threaded_map do |page| retryable(:tries => 3, :on => Twitter::Error::ServerError, :sleep => 0) do client.user_search(query, :page => page, :per_page => MAX_USERS_PER_REQUEST) end end.flatten print_users(users) end end end