Флаттер (Дарт 2) доступ к твиттеру GET api
У меня проблемы с вызовом API Twitter в моем приложении Flutter. Я использовал щебетать в библиотеке, чтобы получить свой токен и секретный токен, и у меня есть мой потребитель и его секрет. Но я не могу правильно сформировать запрос https. Я пытался использовать клиент Oauth2, а также прямой запрос, и ни один из них не сработал.
Я нашел этот репозиторий с реализацией Twitter и Oauth в dart 1, но мне не удалось выяснить, как преобразовать его в Dart 2. Мы очень ценим любую помощь.
Редактировать: вот последний код:
final response = await http.get(new Uri.https(
"api.twitter.com", "/1.1/statuses/home_timeline.json", {
"count": "200",
"tweet_mode": "extended",
"exclude_replies": "false"
}), headers: {
"Authorization": 'Bearer ${twitter.token}', //twitter.token is the token recieved from Twitter sign in process
"Content-Type": "application/json"
});
возвращается "errors":[{"code":89,"message":"Invalid or expired token."}]
Я знаю, что токен действителен
Изменить 2:
Future<List<Tweet>> getTimeline() async {
print("Getting timeline");
var query = https.get(
"https://api.twitter.com/1.1/statuses/home_timeline.json?count=2&tweet_mode=extended&exclude_replies=false",
headers: {
"Authorization":
'oauth_consumer_key="$_consumerKey", oauth_token="${twitter.token}"',
"Content-Type": "application/json"
}).timeout(Duration(seconds: 15));
print("Before await");
final response = await query;
print("code: ${response.statusCode}");
...
}
После дополнительной отладки возникло возможное нулевое исключение с twitter.token. После исправления я все еще получаю неверные данные авторизации. Я буду продолжать пытаться добавить больше информации в шапку и посмотреть, поможет ли это.
Изменить 3:
Вот мой метод генерации подписи:
static String generateSignature(String method, String base, List<String> sortedItems) {
String sig = '$method&${Uri.encodeComponent(base)}&';
String param = '';
for (int i = 0; i < sortedItems.length; i++) {
if (i == 0)
param = sortedItems[i];
else
param += '&${sortedItems[i]}';
}
sig += Uri.encodeComponent(param);
String key = '${Uri.encodeComponent(_secretKey)}&${Uri.encodeComponent(twitter.secret)}';
var digest = Hmac(sha1, utf8.encode(key)).convert(utf8.encode(sig));
print("base: ${digest.bytes}");
print("sig: ${base64.encode(digest.bytes)}");
return base64.encode(digest.bytes);
}
Вот метод временной шкалы:
Future<List<Tweet>> getTimeline() async {
print("Getting timeline");
Future<http.Response> query;
try {
String base = 'https://api.twitter.com/1.1/statuses/home_timeline.json';
String count = 'count=2';
String mode = 'tweet_mode=extended';
String replies = 'exclude_replies=false';
String oauthConsumer = 'oauth_consumer_key="$_consumerKey"';
String oauthToken = 'oauth_token="${twitter.token}"';
String oauthNonce = 'oauth_nonce="${randomAlphaNumeric(20)}"';
String oauthVersion = 'oauth_version="1.0"';
String oauthTime =
'oauth_timestamp="${DateTime.now().millisecondsSinceEpoch}"';
String oauthMethod = 'oauth_signature_method="HMAC-SHA1"';
String oauthSig = 'oauth_signature="${generateSignature("GET", base, [
count,
replies,
oauthConsumer,
oauthNonce,
oauthTime,
oauthToken,
oauthVersion,
mode
])}"';
query = http.get(
new Uri.https("api.twitter.com", "/1.1/statuses/home_timeline.json", {
"count": "2",
"tweet_mode": "extended",
"exclude_replies": "false"
}),
headers: {
"Authorization": '$oauthConsumer, $oauthToken, $oauthVersion, $oauthTime, $oauthNonce, $oauthMethod, $oauthSig',
"Content-Type": "application/json"
}).timeout(Duration(seconds: 15));
} catch (e) {
print(e);
}
Спасибо!
3 ответа
Вот код, который закончил работать:
Метод генерации строки:
static String generateSignature(
String method, String base, List<String> sortedItems) {
String param = '';
for (int i = 0; i < sortedItems.length; i++) {
if (i == 0)
param = sortedItems[i];
else
param += '&${sortedItems[i]}';
}
String sig =
'$method&${Uri.encodeComponent(base)}&${Uri.encodeComponent(param)}';
String key =
'${Uri.encodeComponent(_secretKey)}&${Uri.encodeComponent(twitter.secret)}';
var digest = Hmac(sha1, utf8.encode(key)).convert(utf8.encode(sig));
return base64.encode(digest.bytes);
}
Удобный метод для твиттер-звонков:
Future<http.Response> _twitterGet(
String base, List<List<String>> params) async {
if (twitter == null) await _startSession();
String oauthConsumer =
'oauth_consumer_key="${Uri.encodeComponent(_consumerKey)}"';
String oauthToken = 'oauth_token="${Uri.encodeComponent(twitter.token)}"';
String oauthNonce =
'oauth_nonce="${Uri.encodeComponent(randomAlphaNumeric(42))}"';
String oauthVersion = 'oauth_version="${Uri.encodeComponent("1.0")}"';
String oauthTime =
'oauth_timestamp="${(DateTime.now().millisecondsSinceEpoch / 1000).toString()}"';
String oauthMethod =
'oauth_signature_method="${Uri.encodeComponent("HMAC-SHA1")}"';
var oauthList = [
oauthConsumer.replaceAll('"', ""),
oauthNonce.replaceAll('"', ""),
oauthMethod.replaceAll('"', ""),
oauthTime.replaceAll('"', ""),
oauthToken.replaceAll('"', ""),
oauthVersion.replaceAll('"', "")
];
var paramMap = Map<String, String>();
for (List<String> param in params) {
oauthList.add(
'${Uri.encodeComponent(param[0])}=${Uri.encodeComponent(param[1])}');
paramMap[param[0]] = param[1];
}
oauthList.sort();
String oauthSig =
'oauth_signature="${Uri.encodeComponent(generateSignature("GET", "https://api.twitter.com$base", oauthList))}"';
return await http
.get(new Uri.https("api.twitter.com", base, paramMap), headers: {
"Authorization":
'Oauth $oauthConsumer, $oauthNonce, $oauthSig, $oauthMethod, $oauthTime, $oauthToken, $oauthVersion',
"Content-Type": "application/json"
}).timeout(Duration(seconds: 15));
}
Пример вызова:
Future<User> getUser(String tag) async {
String base = '/1.1/users/show.json';
final response = await _twitterGet(base, [
["screen_name", tag],
["tweet_mode", "extended"]
]);
if (response.statusCode == 200) {
try {
return User(json.decode(response.body));
} catch (e) {
print(e);
return null;
}
} else {
print("Error retrieving user");
print(response.body);
return null;
}
}
Вот документация для аутентификации пользователя в твиттере: https://developer.twitter.com/en/docs/basics/authentication/overview/3-legged-oauth
Первые 3 шага обрабатываются flutter_twitter_login. Взгляните на последний пример:
Step 4: Using these credentials for app-user required requests
Example POST statuses/update
Request includes:
POST statuses/update.json
oauth_consumer_key=cChZNFj6T5R0TigYB9yd1w
oauth_token=7588892-kagSNqWge8gB1WwE3plnFsJHAZVfxWD7Vb57p0b4
Другой пример можно найти здесь: https://developer.twitter.com/en/docs/tweets/post-and-engage/api-reference/post-statuses-update
$ curl --request POST
--url 'https://api.twitter.com/1.1/statuses/update.json?
status=Test%20tweet%20using%20the%20POST%20statuses%2Fupdate%20endpoint'
--header 'authorization: OAuth oauth_consumer_key="YOUR_CONSUMER_KEY",
oauth_nonce="AUTO_GENERATED_NONCE", oauth_signature="AUTO_GENERATED_SIGNATURE",
oauth_signature_method="HMAC-SHA1", oauth_timestamp="AUTO_GENERATED_TIMESTAMP",
oauth_token="USERS_ACCESS_TOKEN", oauth_version="1.0"'
--header 'content-type: application/json'
Я не уверен, что вам нужны все эти аргументы, но заголовок для простого запроса get, вероятно, нуждается как минимум в следующем:
headers: {
"Authorization": 'oauth_consumer_key="${consumerKey}", oauth_token="${twitter.token}"',
"Content-Type": "application/json"
}
По-прежнему не могу заставить его работать.
static String generateSignature(
String method, String base, List<String> sortedItems) {
String sig = '$method&${Uri.encodeComponent(base)}&';
String param = '';
for (int i = 0; i < sortedItems.length; i++) {
if (i == 0)
param = sortedItems[i];
else
param += '&${sortedItems[i]}';
}
sig += Uri.encodeComponent(param);
String key =
'${Uri.encodeComponent("CONSUMERKEY")}&${Uri.encodeComponent("CONSUMERSECRET")}';
var digest = Hmac(sha1, utf8.encode(key)).convert(utf8.encode(sig));
print("base: ${digest.bytes}");
print("sig: ${base64.encode(digest.bytes)}");
return base64.encode(digest.bytes);
}
Future<Null> getTimeline() async {
print("Getting timeline");
Future<http.Response> query;
try {
String base = 'https://api.twitter.com/1.1/statuses/home_timeline.json';
String count = 'count=2';
String mode = 'tweet_mode=extended';
String replies = 'exclude_replies=false';
String oauthConsumer = 'oauth_consumer_key="CONSUMERKEY"';
String oauthToken =
'oauth_token="CONSUMERSECRET"';
String oauthNonce = 'oauth_nonce="${randomAlphaNumeric(20)}"';
String oauthVersion = 'oauth_version="1.0"';
String oauthTime =
'oauth_timestamp="${DateTime.now().millisecondsSinceEpoch}"';
String oauthMethod = 'oauth_signature_method="HMAC-SHA1"';
String oauthSig = 'oauth_signature="${generateSignature("GET", base, [
count,
replies,
oauthConsumer,
oauthNonce,
oauthTime,
oauthToken,
oauthVersion,
mode
])}"';
query = http.get(
new Uri.https("api.twitter.com", "/1.1/statuses/home_timeline.json", {
"count": "2",
"tweet_mode": "extended",
"exclude_replies": "false"
}),
headers: {
"Authorization":
'$oauthConsumer, $oauthToken, $oauthVersion, $oauthTime, $oauthNonce, $oauthMethod, $oauthSig',
"Content-Type": "application/json"
}).timeout(Duration(seconds: 15));
var result = await query;
print(result.body);
} catch (e) {
print(e);
}
}
Что я упустил?