RestSharp和Twitter API 1.1:如何正确地对状态更新文本进行URLEncode?

9
我正在尝试使用RestSharp从我的Web应用程序发布Twitter状态。以下代码完美地运行:
var status = "I am fine with posting this status.";

var client = new RestClient("https://api.twitter.com");

// The OAuth keys/tokens/secrets are retrieved elsewhere
client.Authenticator = OAuth1Authenticator.ForProtectedResource(
    _consumerKey, _consumerSecret, _accessToken, _accessTokenSecret
);

var request = new RestRequest("/1.1/statuses/update.json", Method.POST);
request.AddParameter("status", status, ParameterType.GetOrPost);

var response = client.Execute(request);

然而,如果我在状态文本中包含以下任何字符:! * ' ( ),则此代码会出现身份验证错误。
通过大量的论坛搜寻,我推断出这与OAuth签名编码不匹配POST参数的编码有关。我在这个SO问题中发现了一些信息,但在GitHub上搜索RestSharp问题并没有找到有用的内容。
我可以看到RestSharp源代码中的一些代码UrlEncodeRelaxed),它似乎是手动编码特定字符集以符合OAuth编码规范,因此我尝试以相同方式手动编码我的状态中的这些字符(使用取自RestSharp的代码)后再传递它,例如:
var status = "I'm NOT fine with posting this status.";

string[] UriRfc3986CharsToEscape = new[] { "!", "*", "'", "(", ")" };
string[] UriRfc3968EscapedHex = new[] { "%21", "%2A", "%27", "%28", "%29" };

for (var i = 0; i < UriRfc3986CharsToEscape.Length; i++)
    status = status.Replace(UriRfc3986CharsToEscape[i], UriRfc3968EscapedHex[i]);

但这也不起作用(我仍然收到身份验证错误)。
实际上问题是什么,我应该如何正确编码状态?或者这是RestSharp的一个bug吗?

你找到解决方案或变通方法了吗?我现在正在尝试解决这个问题。 - Kaizen Programmer
@Michael_B 不好意思,我开始研究它,但在其他优先事项出现之前并没有取得太大进展。 - Mark Bell
我最终使用Linq2Twitter进行状态更新。它与Twitter API的v1.1版本兼容。我从未找到解决RestSharp OAuth问题的方法。 - Kaizen Programmer
1个回答

4

您尝试过使用.NET Framework内置的HttpUtility类吗?您可以在System.Web命名空间中找到它。

string urlEncodedText = HttpUtility.UrlEncode("Your text goes here");

MSDN


这明显是一个RestSharp的问题...我尝试解决它一段时间,问题在于没有办法禁用标准的百分数编码...

void Main()
{
    var ProtectedChars = "0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F, 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7A".Replace(" 0x", "%").Split(',');
    var client = new RestClient("https://api.twitter.com");

    client.Authenticator = OAuth1Authenticator.ForProtectedResource(
        "_consumerKey", "_consumerSecret", "_accessToken", "_accessTokenSecret"
    );

    var status = "Dogs, Cats & Mice";

    string newStatus = string.Empty;

    foreach (char c in status)
    {
        var charBytes = Encoding.UTF8.GetBytes(c.ToString());
        var tempText = string.Empty;

        for (int i = 0; i < charBytes.Count(); i++)
        {
            byte b = charBytes[i];
            string hex = "%" + b.ToString("X2");
            tempText += hex;
        }

        if (ProtectedChars.Any(x => x == tempText))
        {
            newStatus += c;
        }
        else
        {
            newStatus += string.Format("{0:X2}", tempText);
        }
    }

    var request = new RestRequest("/1.1/statuses/update.json", Method.POST);
    request.AddParameter(new Parameter{ Name = "status", Type = ParameterType.GetOrPost, Value = newStatus } );

    var response = client.Execute(request);


}

在 Fiddler 中,我一直监控着发送到 Twitter 的请求……这就是我发现的内容……

POST https://api.twitter.com/1.1/statuses/update.json HTTP/1.1
Authorization: /* auth data */
Accept: application/json, application/xml, text/json, text/x-json, text/javascript, text/xml
User-Agent: RestSharp/104.4.0.0
Content-Type: application/x-www-form-urlencoded
Host: api.twitter.com
Content-Length: 44
Accept-Encoding: gzip, deflate

status=Dogs%252C%2520Cats%2520%2526%2520Mice

Http请求的问题在于请求体...如果我提供的值没有被更改...那么Twitter将会识别这条消息并更新我们的状态....

status=Dogs%252C%2520Cats%2520%2526%2520Mice  <--- Altered by RestSharp
status=Dogs%2C%20Cats%20%26%20Mice            <--- What I initially sent

问题在于任何编码方法都无法使其工作:无论是内置的、手动的还是其他方式。这让我相信这是一个内部RestSharp的问题。尽管如此,我仍然没有解决它... - Mark Bell
我刚刚检查了一切...是的...这是一个RestSharp问题...我已经更新了我的答案来解释发生了什么... - Aydin
很有用,谢谢 - 我将查看 RestSharp 的源代码并尝试找出解决方法。 - Mark Bell
尝试使用断点对status字符串进行封装/设置器,希望这能直接引导您找到更改该值的方法。 - Aydin

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接