Я довольно знаком с python и знаю только основы R; поэтому для класса, который требует «использования R», я сильно опираюсь на библиотеку «reticulate».
Я использовал это несколько раз за последний месяц или два без проблем; однако сегодня я определил класс. Я создал экземпляр класса без проблем, но когда я попытался вызвать метод, он вернул ошибку AttributeError: 'TweetGrabber' object has no attribute 'user_search'
Я разобью свой код на то, что работает, а что нет, начиная с работы:
library('reticulate')
## See the below link to download Python if NOT installed locally.
# https://www.anaconda.com/distribution/
py_config()
use_python(python = '/usr/local/bin/python3')
py_available()
py_install("tweepy")
### === Starts Python environment within R! ===
repl_python()
class TweetGrabber(): # Wrapper for Twitter API.
def __init__(self):
import tweepy
self.tweepy = tweepy
myApi = 'my_key'
sApi = 'my_s_key'
at = 'my_at'
sAt = 'my_s_at'
auth = tweepy.OAuthHandler(myApi, sApi)
auth.set_access_token(at, sAt)
self.api = tweepy.API(auth)
def strip_non_ascii(self,string):
''' Returns the string without non ASCII characters'''
stripped = (c for c in string if 0 < ord(c) < 127)
return ''.join(stripped)
def keyword_search(self,keyword,csv_prefix):
import csv
API_results = self.api.search(q=keyword,rpp=1000,show_user=True)
with open(f'{csv_prefix}.csv', 'w', newline='') as csvfile:
fieldnames = ['tweet_id', 'tweet_text', 'date', 'user_id', 'follower_count',
'retweet_count','user_mentions']
writer = csv.DictWriter(csvfile, fieldnames=fieldnames)
writer.writeheader()
for tweet in API_results:
text = self.strip_non_ascii(tweet.text)
date = tweet.created_at.strftime('%m/%d/%Y')
writer.writerow({
'tweet_id': tweet.id_str,
'tweet_text': text,
'date': date,
'user_id': tweet.user.id_str,
'follower_count': tweet.user.followers_count,
'retweet_count': tweet.retweet_count,
'user_mentions':tweet.entities['user_mentions']
})
def user_search(self,user,csv_prefix):
import csv
API_results = self.tweepy.Cursor(self.api.user_timeline,id=user).items()
with open(f'{csv_prefix}.csv', 'w', newline='') as csvfile:
fieldnames = ['tweet_id', 'tweet_text', 'date', 'user_id', 'user_mentions', 'retweet_count']
writer = csv.DictWriter(csvfile, fieldnames=fieldnames)
writer.writeheader()
for tweet in API_results:
text = self.strip_non_ascii(tweet.text)
date = tweet.created_at.strftime('%m/%d/%Y')
writer.writerow({
'tweet_id': tweet.id_str,
'tweet_text': text,
'date': date,
'user_id': tweet.user.id_str,
'user_mentions':tweet.entities['user_mentions'],
'retweet_count': tweet.retweet_count
})
t = TweetGrabber() # Instantiates the class we've designed
Эта следующая строка вызывает ошибку.
t.user_search(user='Telsa',csv_prefix='tesla_tweets') # Find and save to csv Tesla tweets
Заметьте, я запустил этот код в python, и он работает как шарм. Цель - просто простая обёртка API (для обёртки tweepy API), чтобы я мог захватывать и хранить твиты в csv с 1 строкой кода.
Мне известно, что в мире R существуют API-интерфейсы Twitter. Я работаю на сжатой временной шкале, где я пытаюсь избежать изучения twitteR, если только это не единственный вариант. Если это действительно проблема, я могу удалить архитектуру класса и вызывать функции без проблем.
Я озадачен, почему reticulate может так много обрабатывать, не дотягивая до выполнения методов класса. Есть ли проблема в моем коде? Имеет ли это go сверх того, что предназначено для Reticulate?