c# - 当我在查询字符串参数中发送逗号时,twitter api“更新/状态”返回 401
问题描述
我想在 C#、.Net Core 3.1 中使用 Twitter api 发送帖子。我可以使用带有 OAuth 1.0 身份验证的“更新/状态”twitter api 成功发送帖子。但是当我在查询字符串中写逗号时,api 返回 401。
- https://api.twitter.com/1.1/statuses/update.json?status=test1 test2
- https://api.twitter.com/1.1/statuses/update.json?status=test1 , test2
例如,第一个请求成功运行,但第二个请求没有。唯一的区别是逗号。我正在使用 Uri.EscapeDataString()。但它不起作用。我错过了什么?请帮我。谢谢你。
public class OAuth1Manager : IOAuth1Service
{
private OAuth1Request _request;
private List<string> _requestParameters;
private List<string> _signatureParameters;
public List<string> GetQueryStrings(string url)
{
var splitted = url.Split("?");
if (splitted.Length > 1)
{
return splitted[1].Split("&").ToList();
}
return new string[0].ToList();
}
public string GetSignedOAuthString(OAuth1Request request)
{
_request = request;
_requestParameters = new List<string>
{
$"oauth_consumer_key={_request.OAuth.ConsumerKey}",
$"oauth_signature_method=HMAC-SHA1",
$"oauth_timestamp={GetTimeStamp()}",
$"oauth_nonce={GetNonce()}",
$"oauth_token={request.OAuth.AccessToken}",
$"oauth_version=1.0",
};
_signatureParameters = new List<string>();
_signatureParameters.AddRange(_requestParameters);
_signatureParameters.AddRange(GetQueryStrings(_request.Url));
_requestParameters.Add($"oauth_signature={GetSignature()}");
for (int i = 0; i < _requestParameters.Count; i++)
{
var parameter = _requestParameters[i];
var splitted = parameter.Split("=");
_requestParameters[i] = $"{splitted[0]}=\"{splitted[1]}\"";
}
return $"OAuth {_requestParameters.JoinAsString(",")}";
}
private string GetNonce()
{
var rand = new Random();
var nonce = rand.Next(1000000000);
return nonce.ToString();
}
private string GetSignatureBaseString()
{
var sortedList = new List<string>(_signatureParameters);
sortedList.Sort();
var requestParametersSortedString = sortedList.JoinAsString("&");
var url = ConstructRequestUrl();
return _request.Method.GetEnumDescription().ToUpper() + "&" + Uri.EscapeDataString(url) + "&" +
Uri.EscapeDataString(requestParametersSortedString);
}
private string ConstructRequestUrl()
{
var uri = new Uri(_request.Url, UriKind.Absolute);
var normUrl = string.Format("{0}://{1}", uri.Scheme, uri.Host);
if (!(uri.Scheme == "http" && uri.Port == 80 ||
uri.Scheme == "https" && uri.Port == 443))
normUrl += ":" + uri.Port;
normUrl += uri.AbsolutePath;
return normUrl;
}
private string GetSignature()
{
var hmacsha1 = new HMACSHA1();
var key = Uri.EscapeDataString(_request.OAuth.ConsumerSecret) + "&" + (string.IsNullOrEmpty(_request.OAuth.TokenSecret)
? ""
: Uri.EscapeDataString(_request.OAuth.TokenSecret));
hmacsha1.Key = Encoding.ASCII.GetBytes(key);
var dataBuffer = Encoding.ASCII.GetBytes(GetSignatureBaseString());
var hashBytes = hmacsha1.ComputeHash(dataBuffer);
return Uri.EscapeDataString(Convert.ToBase64String(hashBytes));
}
private string GetTimeStamp()
{
var ts = DateTime.UtcNow - new DateTime(1970, 1, 1, 0, 0, 0, 0);
return Convert.ToInt64(ts.TotalSeconds).ToString();
}
}
解决方案
我用 %2C 替换了所有逗号,比如“test1%2Ctest2”而不是“test1,test2”,然后它就起作用了。
推荐阅读
- wordpress - 我究竟做错了什么?我正在尝试使用 wp all import 导入 woocommerce
- android - 如何在加载插页式广告之前通知用户
- phpmyadmin - 您无权访问此服务器上的 /phpmyadmin/。在 centos 32bit / apache 2.2.15
- php - 如何在刀片 laravel 的两种情况下使用 where
- python - 更改用户内联后,如何在 Django 管理员中发送一次电子邮件?
- node.js - 打字稿函数返回两个不同的对象
- javascript - 将基于偏好的对象合并为一个
- javascript - 如何在反应组件中使用 exif-js?
- sql - 明智地显示员工表部门的运行工资,而不使用滞后、领先或分区
- android - Firebase 从特定子值的多个键更新多个子值