использование TwitPic + OAuth для загрузки фото + твита в Twitter (.NET C#) — почему нет твита?

Я загружаю фотографии в TwitPic с помощью OAuth из приложения .NET, написанного на C#.

Материал oAuth немного сложен. Я нашел два фрагмента кода .NET, чтобы справиться с этим, но ни один из них меня не удовлетворил. DotNetOpenAuth оказался довольно тяжелым, больше, чем мне нужно. (Просто хочу делать подписи oAuth и запросы токенов). Код OAuthBase.cs показался мне запутанным и неэлегантным. Мне пришлось передать в методы 6 строковых параметров, и если какой-то из них окажется не по порядку, горе мне.

Поэтому я написал некоторый код, чтобы сделать это самостоятельно, он довольно маленький и, кажется, работает. Он работает для получения «токенов запроса». Он работает, чтобы открыть веб-страницу авторизации и получить «токены доступа». Он также работает для загрузки фотографий в TwitPic.

Все ответы HTTP возвращаются как 200 или 201.

загрузить HTTP-сообщение выглядит следующим образом:

POST http://api.twitpic.com/2/upload.json HTTP/1.1
Content-Type: multipart/form-data; boundary=48cb9a6d-1f1d-432d-b6e3-307e32e8228a
X-Auth-Service-Provider: https://api.twitter.com/1/account/verify_credentials.json
X-Verify-Credentials-Authorization: OAuth realm="http://api.twitter.com/",
  oauth_consumer_key="Dv1er93yKzEMn74hZfPmJA",
  oauth_nonce="51fi305k",
  oauth_signature="4oWcmZcd%2F%2F81JslJ70xFXFm8%2BQs%3D",
  oauth_signature_method="HMAC-SHA1",
  oauth_timestamp="1292277715",
  oauth_token="59152613-z8EP4GoYS1Mgo3E29JfIqBnyTRlruAJs8Bkvs3q0T",
  oauth_version="1.0"
Host: api.twitpic.com
Content-Length: 14605
Expect: 100-continue
Connection: Keep-Alive

--48cb9a6d-1f1d-432d-b6e3-307e32e8228a
Content-Disposition: file; name="media"; filename="CropperCapture[10].jpg"
Content-Type: image/jpeg
....
--48cb9a6d-1f1d-432d-b6e3-307e32e8228a
Content-Disposition: form-data; name="key"

twitpic-api-key-here
--48cb9a6d-1f1d-432d-b6e3-307e32e8228a
Content-Disposition: form-data; name="message"

uploaded from Yappy. (at 12/13/2010 5:01:55 PM)
--48cb9a6d-1f1d-432d-b6e3-307e32e8228a--

Json, который я получаю после загрузки, выглядит следующим образом:

{"id":"3f0jeiw5",
 "text":"uploaded from Yappy. (at 12\/13\/2010 5:01:55 PM)",
 "url":"http:\\/twitpic.com\/3f0jeiw5",
 "width":257,
 "height":184,
 "size":14156,
 "type":"jpg",
 "timestamp":"Mon, 13 Dec 2010 22:02:06 +0000",
 "user":{
   "id":54591561,"screen_name":"bfavre"}
}

Но проблема в том, что после того, как я загрузил изображение на Twitpic, изображение доступно на TwitPic, но связанное с ним сообщение никогда не появляется в Twitter.

Что дает?

Я прочитал в случайный пост в блоге, что использование TwitPic+oAuth требует, чтобы я разместил твит-сообщение в отдельной HTTP-транзакции, прямо в Twitter. Хм? Я думал, что цель oAuth для почты — позволить потребителям делать что-то от моего имени — например, разрешить TwitPic публиковать твиты для меня.

Любые подсказки?


ИЗМЕНИТЬ
Здесь я узнаю немного больше. Это сообщение в блоге от мая 2010 г. предлагает мне использовать значение для X-Auth-Service-Provider из https://api.twitter.com/1/account/verify_credentials.json говорит TwitPic вызывать «verify_credentials.json» на twitter.com, когда он получает мой запрос. Если это действительно просто проверка моих учетных данных, это объясняет, почему твиты не публикуются.

В сообщении также предлагается заменить его на https://api.twitter.com/1/status/update.json должен позволить мне обновлять Twitter через TwitPic с делегированием. Но это дальновидный пост — в нем говорится, что получение этой возможности требует работы со стороны Twitter.

Я еще не нашел примера HTTP-сообщения, которое делает это. Кто-нибудь?


ОБНОВЛЕНИЕ
После преобразования URL-адреса подтверждения в https://api.twitter.com/1/status/update.json и использования POST для подписи я получаю код ответа 401:

{"errors":
  [{"code":401,
    "message":"Could not authenticate you (header rejected by twitter)."}]
}

По сути, это та же проблема, которая описана здесь, на форуме разработчиков Twitter. . Предположение в конце этой темы заключалось в том, что алгоритм вычисления подписи неверен, но я думаю, что это неправильно, поскольку мой алгоритм подписи работает со всеми другими запросами.


person Cheeso    schedule 13.12.2010    source источник


Ответы (2)


Я работал над использованием Twitpic API для создания интеграции oAuth и обнаружил, что могу опубликовать изображение и отправить комментарий в Twitter вместе с изображением из Twitpic. Это отправляет изображение в учетную запись пользователя Twitter нашего видеочата Eplixo (http://eplixo.com/m/) и клиента Twitter.

Однако я не могу получить ответ. Теперь я могу жить с его публикацией и загрузкой, но было бы полезно выяснить, как получить данные ответа для других частей приложения.

Вот что у меня есть. Это вариант оболочки Twipli API.

protected void Button1_Click(object sender, EventArgs e)
{

    string ct = img.PostedFile.ContentType.ToString();
    string usertoken = Session["usrToken"].ToString();
    string userSecret = Session["usrSecret"].ToString();
    string conkey = Session["ConsumerKey"].ToString();
    string consecret = Session["ConsumerSecret"].ToString();
    string twitkey = Session["twitpickey"].ToString();

    string _m = m.Text; // This takes the Tweet to be posted


    HttpPostedFile myFile = img.PostedFile;
    string fileName = myFile.FileName.ToString();

    int nFileLen = myFile.ContentLength;
    byte[] myData = new byte[nFileLen];
    myFile.InputStream.Read(myData, 0, nFileLen);

    TwitPic tw = new TwitPic();
    upres.Text = tw.UploadPhoto(myData, ct, _m, fileName, twitkey, usertoken, userSecret, conkey, consecret).ToString();
    Response.Redirect("twittercb.aspx?oauth_verifier=none");
}
public class TwitPic
{
    private const string TWITPIC_UPLADO_API_URL = "http://api.twitpic.com/2/upload";
    private const string TWITPIC_UPLOAD_AND_POST_API_URL = "http://api.twitpic.com/1/uploadAndPost";
    /// 
    /// Uploads the photo and sends a new Tweet
    /// 
    /// <param name="binaryImageData">The binary image data.
    /// <param name="tweetMessage">The tweet message.
    /// <param name="filename">The filename.
    /// Return true, if the operation was succeded.
    public string UploadPhoto(byte[] binaryImageData, string ContentType, string tweetMessage, string filename, string tpkey, string usrtoken, string usrsecret, string contoken, string consecret)
    {            
        string boundary = Guid.NewGuid().ToString();
        string requestUrl = String.IsNullOrEmpty(tweetMessage) ? TWITPIC_UPLADO_API_URL : TWITPIC_UPLOAD_AND_POST_API_URL;
        HttpWebRequest request = (HttpWebRequest)WebRequest.Create(requestUrl);
        string encoding = "iso-8859-1";

        request.PreAuthenticate = true;
        request.AllowWriteStreamBuffering = true;
        request.ContentType = string.Format("multipart/form-data; boundary={0}", boundary);
        request.Method = "POST";

        string header = string.Format("--{0}", boundary);
        string footer = string.Format("--{0}--", boundary);

        StringBuilder contents = new StringBuilder();
        contents.AppendLine(header);

        string fileContentType = ContentType;
        string fileHeader = String.Format("Content-Disposition: file; name=\"{0}\"; filename=\"{1}\"", "media", filename);
        string fileData = Encoding.GetEncoding(encoding).GetString(binaryImageData);

        contents.AppendLine(fileHeader);
        contents.AppendLine(String.Format("Content-Type: {0}", fileContentType));
        contents.AppendLine();
        contents.AppendLine(fileData);

        contents.AppendLine(header);
        contents.AppendLine(String.Format("Content-Disposition: form-data; name=\"{0}\"", "key"));
        contents.AppendLine();
        contents.AppendLine(tpkey);

        contents.AppendLine(header);
        contents.AppendLine(String.Format("Content-Disposition: form-data; name=\"{0}\"", "consumer_token"));
        contents.AppendLine();
        contents.AppendLine(contoken);

        contents.AppendLine(header);
        contents.AppendLine(String.Format("Content-Disposition: form-data; name=\"{0}\"", "consumer_secret"));
        contents.AppendLine();
        contents.AppendLine(consecret);

        contents.AppendLine(header);
        contents.AppendLine(String.Format("Content-Disposition: form-data; name=\"{0}\"", "oauth_token"));
        contents.AppendLine();
        contents.AppendLine(usrtoken);

        contents.AppendLine(header);
        contents.AppendLine(String.Format("Content-Disposition: form-data; name=\"{0}\"", "oauth_secret"));
        contents.AppendLine();
        contents.AppendLine(usrsecret);

        if (!String.IsNullOrEmpty(tweetMessage))
        {
            contents.AppendLine(header);
            contents.AppendLine(String.Format("Content-Disposition: form-data; name=\"{0}\"", "message"));
            contents.AppendLine();
            contents.AppendLine(tweetMessage);
        }

        contents.AppendLine(footer);            
        byte[] bytes = Encoding.GetEncoding(encoding).GetBytes(contents.ToString());            
        request.ContentLength = bytes.Length;

        string mediaurl = "";
        try
        {
            using (Stream requestStream = request.GetRequestStream()) // this is where the bug is due to not being able to seek.
            {        
                requestStream.Write(bytes, 0, bytes.Length); // No problem the image is posted and tweet is posted
                requestStream.Close();                       
                using (HttpWebResponse response = (HttpWebResponse)request.GetResponse()) // here I can't get the response
                { 
                    using (StreamReader reader = new StreamReader(response.GetResponseStream()))
                    {
                        string result = reader.ReadToEnd();

                        XDocument doc = XDocument.Parse(result); // this shows no root elements and fails here

                        XElement rsp = doc.Element("rsp");
                        string status = rsp.Attribute(XName.Get("status")) != null ? rsp.Attribute(XName.Get("status")).Value : rsp.Attribute(XName.Get("stat")).Value;
                        mediaurl = rsp.Element("mediaurl").Value;
                        return mediaurl;                            
                    } 
                } 

            }
        }
        catch (Exception ex)
        {
            ex.ToString();
        } 
        return mediaurl;
    }

}
person Gerard Brandon    schedule 24.12.2010

Я не знаю окончательного ответа, но я думаю, что изменения, которые были описаны как «Очень скоро сейчас» в запись в блоге Раффи от мая 2010 года по какой-то причине так и не была опубликована.

Так что на самом деле, используя OAuth, вам нужно публиковать отдельно в TwitPic и Twitter.

Однако сделать это в Twitter несложно. Просто выполните POST для URL-адреса statuses/update.xml.

private void Tweet(string message)
{
    var twitterUpdateUrlBase = "http://api.twitter.com/1/statuses/update.xml?status=";
    var url = twitterUpdateUrlBase + UrlEncode(message);

    var authzHeader = oauth.GenerateAuthzHeader(url, "POST");

    var request = (HttpWebRequest)WebRequest.Create(url);
    request.Method = "POST";
    request.PreAuthenticate = true;
    request.AllowWriteStreamBuffering = true;
    request.Headers.Add("Authorization", authzHeader);

    using (var response = (HttpWebResponse)request.GetResponse())
    {
        if (response.StatusCode != HttpStatusCode.OK)
        MessageBox.Show("There's been a problem trying to tweet:" +
                        Environment.NewLine +
                        response.StatusDescription +
                        Environment.NewLine +
                        Environment.NewLine +
                        "You will have to tweet manually." +
                        Environment.NewLine);
    }
}

В этом коде есть две сложные части: первая — это вызов UrlEncode(). OAuth указывает, что URL-кодирование должно использовать заглавные буквы. Встроенная подпрограмма .NET использует строчные буквы. Так что обязательно делайте апкейс.

Вторая сложная часть — это получение заголовка авторизации OAuth. Если вы используете пакет библиотеки OAuth, это должно быть довольно просто. Если вам нужен простой, вы можете получить его здесь: OAuth.cs. (Загрузите OAuthManager)

person Cheeso    schedule 14.12.2010
comment
К сведению: мне нравится ваша простая библиотека OAuth. Я сделал быстрое исправление для обработки расширенных символов и разместил его здесь: cropperplugins.codeplex.com/discussions /249415 - person russau; 14.03.2011
comment
Просто продолжаю - спасибо, ваши изменения теперь в библиотеке OAuth; Кроме того, теперь есть DLL, которую может скачать любой желающий. - person Cheeso; 25.04.2011