trabajo-afectivo/app/models/channel/driver/twitter.rb

326 lines
7.8 KiB
Ruby
Raw Normal View History

2015-07-02 15:13:04 +00:00
# Copyright (C) 2012-2015 Zammad Foundation, http://zammad-foundation.org/
2016-01-10 13:24:54 +00:00
class Channel::Driver::Twitter
2015-12-14 09:23:14 +00:00
=begin
fetch tweets from twitter account
options = {
adapter: 'twitter',
auth: {
consumer_key: consumer_key,
consumer_secret: consumer_secret,
oauth_token: armin_theo_token,
oauth_token_secret: armin_theo_token_secret,
},
sync: {
search: [
{
term: '#citheo42',
group_id: 2,
},
{
term: '#citheo24',
group_id: 1,
},
],
mentions: {
group_id: 2,
},
direct_messages: {
group_id: 2,
}
}
}
instance = Channel::Driver::Twitter.new
result = instance.fetch(options, channel)
returns
{
result: 'ok',
}
=end
def fetch (options, channel)
2015-07-02 15:13:04 +00:00
2015-12-21 00:48:49 +00:00
options = check_external_credential(options)
@rest_client = TweetRest.new(options[:auth])
@sync = options[:sync]
@channel = channel
2015-07-02 15:13:04 +00:00
Rails.logger.debug 'twitter fetch started'
2015-07-02 15:13:04 +00:00
fetch_mentions
fetch_search
2015-07-02 15:13:04 +00:00
fetch_direct_messages
disconnect
Rails.logger.debug 'twitter fetch completed'
2015-12-14 09:23:14 +00:00
{
result: 'ok',
notice: '',
2015-12-14 09:23:14 +00:00
}
2015-07-02 15:13:04 +00:00
end
2016-01-10 13:24:54 +00:00
=begin
instance = Channel::Driver::Twitter.new
instance.fetchable?(channel)
=end
def fetchable?(channel)
return true if Rails.env.test?
# only fetch once in 30 minutes
2016-01-10 13:24:54 +00:00
return true if !channel.preferences
return true if !channel.preferences[:last_fetch]
return false if channel.preferences[:last_fetch] > Time.zone.now - 30.minutes
2016-01-10 13:24:54 +00:00
true
end
2015-12-14 09:23:14 +00:00
=begin
2015-07-02 15:13:04 +00:00
2015-12-14 09:23:14 +00:00
instance = Channel::Driver::Twitter.new
instance.send(
{
adapter: 'twitter',
auth: {
consumer_key: consumer_key,
consumer_secret: consumer_secret,
oauth_token: armin_theo_token,
oauth_token_secret: armin_theo_token_secret,
},
},
twitter_attributes,
notification
)
2015-07-02 15:13:04 +00:00
2015-12-14 09:23:14 +00:00
=end
2015-07-02 15:13:04 +00:00
2015-12-14 09:23:14 +00:00
def send(options, article, _notification = false)
# return if we run import mode
return if Setting.get('import_mode')
2015-12-21 00:48:49 +00:00
options = check_external_credential(options)
@rest_client = TweetRest.new(options[:auth])
tweet = @rest_client.from_article(article)
2015-12-14 09:23:14 +00:00
disconnect
2015-07-02 15:13:04 +00:00
tweet
end
def disconnect
@stream_client.disconnect if @stream_client
@rest_client.disconnect if @rest_client
end
2016-01-09 12:23:11 +00:00
=begin
create stream endpoint form twitter account
options = {
adapter: 'twitter',
auth: {
consumer_key: consumer_key,
consumer_secret: consumer_secret,
oauth_token: armin_theo_token,
oauth_token_secret: armin_theo_token_secret,
},
sync: {
search: [
{
term: '#citheo42',
group_id: 2,
},
{
term: '#citheo24',
group_id: 1,
},
],
mentions: {
group_id: 2,
},
direct_messages: {
group_id: 2,
}
}
}
instance = Channel::Driver::Twitter.new
stream_instance = instance.stream_instance(channel)
returns
instance_of_stream_handle
=end
def stream_instance(channel)
@channel = channel
options = @channel.options
@stream_client = TweetStream.new(options[:auth])
end
2016-01-09 12:23:11 +00:00
=begin
stream tweets from twitter account
stream_instance.stream
returns
# endless loop
=end
def stream
sync = @channel.options['sync']
2016-03-01 14:26:46 +00:00
raise 'Need channel.options[\'sync\'] for account, but no params found' if !sync
filter = {}
if sync['search']
hashtags = []
2016-06-30 20:04:48 +00:00
sync['search'].each { |item|
hashtags.push item['term']
}
filter[:track] = hashtags.join(',')
end
if sync['mentions'] && sync['mentions']['group_id'] != ''
filter[:replies] = 'all'
end
return if filter.empty?
@stream_client.client.user(filter) do |tweet|
next if tweet.class != Twitter::Tweet && tweet.class != Twitter::DirectMessage
next if Ticket::Article.find_by(message_id: tweet.id)
# check direct message
if tweet.class == Twitter::DirectMessage
if sync['direct_messages'] && sync['direct_messages']['group_id'] != ''
2016-01-09 12:23:11 +00:00
next if @stream_client.direct_message_limit_reached(tweet)
@stream_client.to_group(tweet, sync['direct_messages']['group_id'], @channel)
end
next
end
next if @stream_client.tweet_limit_reached(tweet)
# check if it's mention
if sync['mentions'] && sync['mentions']['group_id'] != ''
hit = false
if tweet.user_mentions
2016-06-30 20:04:48 +00:00
tweet.user_mentions.each { |user|
if user.id.to_s == @channel.options['user']['id'].to_s
hit = true
end
}
end
if hit
@stream_client.to_group(tweet, sync['mentions']['group_id'], @channel)
next
end
end
# check hashtags
if sync['search'] && tweet.hashtags
hit = false
2016-06-30 20:04:48 +00:00
sync['search'].each { |item|
tweet.hashtags.each { |hashtag|
next if item['term'] !~ /^#/
if item['term'].sub(/^#/, '') == hashtag.text
hit = item
end
}
}
if hit
@stream_client.to_group(tweet, hit['group_id'], @channel)
next
end
end
# check stings
if sync['search']
hit = false
body = tweet.text
2016-06-30 20:04:48 +00:00
sync['search'].each { |item|
next if item['term'] =~ /^#/
if body =~ /#{item['term']}/
hit = item
end
}
if hit
@stream_client.to_group(tweet, hit['group_id'], @channel)
end
end
end
2015-07-02 15:13:04 +00:00
end
private
def fetch_search
return if !@sync[:search]
2015-07-03 06:55:59 +00:00
return if @sync[:search].empty?
2015-07-02 15:13:04 +00:00
@sync[:search].each { |search|
2016-02-11 01:10:23 +00:00
next if !search[:term]
next if search[:term].to_s.empty?
next if !search[:group_id]
next if search[:group_id].to_s.empty?
2015-07-02 15:13:04 +00:00
result_type = search[:type] || 'mixed'
Rails.logger.debug " - searching for '#{search[:term]}'"
@rest_client.client.search(search[:term], result_type: result_type).collect { |tweet|
next if Ticket::Article.find_by(message_id: tweet.id)
break if @rest_client.tweet_limit_reached(tweet)
@rest_client.to_group(tweet, search[:group_id], @channel)
2015-07-02 15:13:04 +00:00
}
}
end
def fetch_mentions
return if !@sync[:mentions]
2015-07-03 06:55:59 +00:00
return if @sync[:mentions].empty?
2016-02-11 01:10:23 +00:00
return if !@sync[:mentions][:group_id]
return if @sync[:mentions][:group_id].to_s.empty?
2015-07-02 15:13:04 +00:00
Rails.logger.debug ' - searching for mentions'
@rest_client.client.mentions_timeline.each { |tweet|
next if Ticket::Article.find_by(message_id: tweet.id)
break if @rest_client.tweet_limit_reached(tweet)
@rest_client.to_group(tweet, @sync[:mentions][:group_id], @channel)
2015-07-02 15:13:04 +00:00
}
end
def fetch_direct_messages
return if !@sync[:direct_messages]
2015-07-03 06:55:59 +00:00
return if @sync[:direct_messages].empty?
2016-02-11 01:10:23 +00:00
return if !@sync[:direct_messages][:group_id]
return if @sync[:direct_messages][:group_id].to_s.empty?
2015-07-02 15:13:04 +00:00
Rails.logger.debug ' - searching for direct_messages'
@rest_client.client.direct_messages.each { |tweet|
next if Ticket::Article.find_by(message_id: tweet.id)
break if @rest_client.direct_message_limit_reached(tweet)
@rest_client.to_group(tweet, @sync[:direct_messages][:group_id], @channel)
2015-07-02 15:13:04 +00:00
}
end
2015-12-21 00:48:49 +00:00
def check_external_credential(options)
if options[:auth] && options[:auth][:external_credential_id]
external_credential = ExternalCredential.find_by(id: options[:auth][:external_credential_id])
2016-03-01 14:26:46 +00:00
raise "No such ExternalCredential.find(#{options[:auth][:external_credential_id]})" if !external_credential
2015-12-21 00:48:49 +00:00
options[:auth][:consumer_key] = external_credential.credentials['consumer_key']
options[:auth][:consumer_secret] = external_credential.credentials['consumer_secret']
end
options
end
2015-07-02 15:13:04 +00:00
end