/* Facebook plugin for Miranda Instant Messenger _____________________________________________ Copyright © 2009-11 Michal Zelinka This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . File name : $HeadURL: http://eternityplugins.googlecode.com/svn/trunk/facebook/connection.cpp $ Revision : $Revision: 91 $ Last change by : $Author: n3weRm0re.ewer $ Last change on : $Date: 2011-01-08 11:10:34 +0100 (so, 08 1 2011) $ */ #include "common.h" void FacebookProto::ChangeStatus(void*) { ScopedLock s(signon_lock_); ScopedLock b(facy.buddies_lock_); int new_status = m_iDesiredStatus; int old_status = m_iStatus; if ( new_status == ID_STATUS_OFFLINE ) { // Logout LOG("##### Beginning SignOff process"); m_iStatus = facy.self_.status_id = ID_STATUS_OFFLINE; SetEvent(update_loop_lock_); Netlib_Shutdown(facy.hMsgCon); if ( getByte(FACEBOOK_KEY_DISCONNECT_CHAT, DEFAULT_DISCONNECT_CHAT) ) facy.chat_state( false ); facy.logout( ); deleteSetting( "LogonTS" ); facy.clear_cookies( ); facy.buddies.clear( ); ProtoBroadcastAck(m_szModuleName, 0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)old_status, m_iStatus); OnLeaveChat(NULL, NULL); SetAllContactStatuses( ID_STATUS_OFFLINE ); ToggleStatusMenuItems(false); if (facy.hMsgCon) Netlib_CloseHandle(facy.hMsgCon); facy.hMsgCon = NULL; LOG("##### SignOff complete"); return; } else if ( old_status == ID_STATUS_OFFLINE ) { // Login SYSTEMTIME t; GetLocalTime( &t ); Log("[%d.%d.%d] Using Facebook Protocol RM %s", t.wDay, t.wMonth, t.wYear, __VERSION_STRING); LOG("***** Beginning SignOn process"); m_iStatus = facy.self_.status_id = ID_STATUS_CONNECTING; ProtoBroadcastAck(m_szModuleName,0,ACKTYPE_STATUS,ACKRESULT_SUCCESS, (HANDLE)old_status, m_iStatus); ResetEvent(update_loop_lock_); if ( NegotiateConnection( ) ) { facy.last_feeds_update_ = ::time( NULL ); facy.home(); facy.reconnect(); facy.load_friends(); if (getByte(FACEBOOK_KEY_PARSE_MESSAGES, DEFAULT_PARSE_MESSAGES)) ForkThread( &FacebookProto::ProcessUnreadMessages, this ); setDword( "LogonTS", (DWORD)time(NULL) ); ForkThread( &FacebookProto::UpdateLoop, this ); ForkThread( &FacebookProto::MessageLoop, this ); if (getByte(FACEBOOK_KEY_SET_MIRANDA_STATUS, DEFAULT_SET_MIRANDA_STATUS)) { ForkThread(&FacebookProto::SetAwayMsgWorker, this, NULL); } } else { ProtoBroadcastAck(m_szModuleName,0,ACKTYPE_STATUS,ACKRESULT_FAILED, (HANDLE)old_status,m_iStatus); // Set to offline m_iStatus = m_iDesiredStatus = facy.self_.status_id = ID_STATUS_OFFLINE; ProtoBroadcastAck(m_szModuleName, 0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)old_status, m_iStatus); LOG("***** SignOn failed"); return; } ToggleStatusMenuItems(true); LOG("***** SignOn complete"); } else if ( new_status == ID_STATUS_INVISIBLE ) { facy.buddies.clear( ); this->SetAllContactStatuses( ID_STATUS_OFFLINE ); } facy.chat_state( m_iDesiredStatus != ID_STATUS_INVISIBLE ); facy.buddy_list( ); m_iStatus = facy.self_.status_id = m_iDesiredStatus; ProtoBroadcastAck(m_szModuleName, 0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)old_status, m_iStatus); LOG("***** ChangeStatus complete"); } /** Return true on success, false on error. */ bool FacebookProto::NegotiateConnection( ) { LOG("***** Negotiating connection with Facebook"); bool error; std::string user, pass; DBVARIANT dbv = {0}; error = true; if ( !DBGetContactSettingString(NULL,m_szModuleName,FACEBOOK_KEY_LOGIN,&dbv) ) { user = dbv.pszVal; DBFreeVariant(&dbv); error = user.empty(); } if (error) { NotifyEvent(m_tszUserName,TranslateT("Please enter a username."),NULL,FACEBOOK_EVENT_CLIENT); return false; } error = true; if ( !DBGetContactSettingString(NULL,m_szModuleName,FACEBOOK_KEY_PASS,&dbv) ) { CallService(MS_DB_CRYPT_DECODESTRING,strlen(dbv.pszVal)+1, reinterpret_cast(dbv.pszVal)); pass = dbv.pszVal; DBFreeVariant(&dbv); error = pass.empty(); } if (error) { NotifyEvent(m_tszUserName,TranslateT("Please enter a password."),NULL,FACEBOOK_EVENT_CLIENT); return false; } // Load machine name if ( !DBGetContactSettingString(NULL,m_szModuleName,FACEBOOK_KEY_DEVICE_ID,&dbv) ) { facy.cookies["datr"] = dbv.pszVal; DBFreeVariant(&dbv); } // Get info about secured connection facy.https_ = DBGetContactSettingByte(NULL, m_szModuleName, FACEBOOK_KEY_FORCE_HTTPS, DEFAULT_FORCE_HTTPS ) != 0; return facy.login( user, pass ); } void FacebookProto::UpdateLoop(void *) { time_t tim = ::time(NULL); LOG( ">>>>> Entering Facebook::UpdateLoop[%d]", tim ); for ( int i = -1; !isOffline(); i = ++i % 6 ) { if ( i != -1 ) { if ( !facy.invisible_ ) if ( !facy.buddy_list( ) ) break; } if ( i == 2 && getByte( FACEBOOK_KEY_EVENT_FEEDS_ENABLE, DEFAULT_EVENT_FEEDS_ENABLE ) ) if ( !facy.feeds( ) ) break; LOG( "***** FacebookProto::UpdateLoop[%d] going to sleep...", tim ); if ( WaitForSingleObjectEx( update_loop_lock_, GetPollRate( ) * 1000, true ) != WAIT_TIMEOUT ) break; LOG( "***** FacebookProto::UpdateLoop[%d] waking up...", tim ); } ResetEvent(update_loop_lock_); LOG( "<<<<< Exiting FacebookProto::UpdateLoop[%d]", tim ); } void FacebookProto::MessageLoop(void *) { //ScopedLock s(message_loop_lock_); // TODO: Required? time_t tim = ::time(NULL); LOG( ">>>>> Entering Facebook::MessageLoop[%d]", tim ); while ( facy.channel( ) ) { if ( isOffline() ) break; LOG( "***** FacebookProto::MessageLoop[%d] refreshing...", tim ); } LOG( "<<<<< Exiting FacebookProto::MessageLoop[%d]", tim ); } BYTE FacebookProto::GetPollRate( ) { BYTE poll_rate = getByte( FACEBOOK_KEY_POLL_RATE, FACEBOOK_DEFAULT_POLL_RATE ); return ( ( poll_rate >= FACEBOOK_MINIMAL_POLL_RATE && poll_rate <= FACEBOOK_MAXIMAL_POLL_RATE ) ? poll_rate : FACEBOOK_DEFAULT_POLL_RATE ); }