/************************************************************************ Copyright (c) 2000 - 2000 Microsoft Corporation Module Name : progressivedl.h Abstract : Main header file for downloader. Author : Revision History : ***********************************************************************/ #pragma once #define MIN(a, b) (a>b ? b:a) #define MAX(a, b) (a>b ? a:b) #define MAX_REPLY_DATA (2 * INTERNET_MAX_URL_LENGTH) #define E_RETRY HRESULT_FROM_WIN32( ERROR_RETRY ) #define BG_E_HEADER_NOT_FOUND HRESULT_FROM_WIN32( ERROR_HTTP_HEADER_NOT_FOUND ) // // valid in HTTP 1.1, but not defined in the Windows-XP version of wininet.h // #define HTTP_STATUS_RANGE_NOT_SATISFIABLE 416 //---------------------------------------------------------------------- typedef HRESULT (*QMCALLBACK)(DWORD, DWORD, LPBYTE, DWORD); #define MAX_VIA_HEADER_LENGTH 300 // // This is the size of the read buffer for downloads. 48kb is dramatically better than 32kb or 64kb, // for obscure reasons. // #define FILE_DATA_BUFFER_LEN (48*1024) extern BYTE g_FileDataBuffer[]; //-------------------------------------------- struct URL_INFO { HINTERNET hInternet; HINTERNET hConnect; FILETIME UrlModificationTime; UINT64 FileSize; DWORD dwFlags; bool bHttp11; bool bRange; bool fProxy; // // Most of these could be stack variables, but they are too big. // TCHAR HostName[INTERNET_MAX_URL_LENGTH + 1]; // host-relative URL // TCHAR UrlPath[INTERNET_MAX_URL_LENGTH + 1]; INTERNET_PORT Port; // a copy of with range information appended as parameters // TCHAR BlockUrl[INTERNET_MAX_URL_LENGTH + 1]; TCHAR UserName[UNLEN + 1]; TCHAR Password[UNLEN + 1]; PROXY_SETTINGS_CONTAINER * ProxySettings; CAutoString ProxyHost; TCHAR ViaHeader[ MAX_VIA_HEADER_LENGTH + 1]; const CCredentialsContainer * Credentials; //-------- URL_INFO( LPCTSTR a_Url, const PROXY_SETTINGS * a_ProxySettings, const CCredentialsContainer * a_Credentials, LPCTSTR HostId = NULL ); ~URL_INFO(); void Cleanup(); QMErrInfo Connect(); void Disconnect(); BOOL GetProxyUsage( HINTERNET hRequest, QMErrInfo *pQMErrInfo ); }; //--------------------------------------------- class CUploadJob; class CJobManager; class ITransferCallback { public: virtual bool DownloaderProgress( UINT64 BytesTransferred, UINT64 BytesTotal ) = 0; virtual bool PollAbort() = 0; virtual bool UploaderProgress( UINT64 BytesTransferred ) = 0; }; class Downloader { public: virtual HRESULT Download( LPCTSTR szURL, LPCTSTR szDest, UINT64 Offset, ITransferCallback *CallBack, QMErrInfo *pErrInfo, HANDLE hToken, BOOL bThrottle, const PROXY_SETTINGS * pProxySettings, // optional const CCredentialsContainer * Credentials, const StringHandle HostId = StringHandle() // optional ) = 0; virtual HRESULT GetRemoteFileInformation( HANDLE hToken, LPCTSTR szURL, UINT64 * pFileSize, FILETIME *pFileTime, QMErrInfo *pErrInfo, const PROXY_SETTINGS * pProxySettings, //optional const CCredentialsContainer * Credentials, // optional const StringHandle HostId = StringHandle() // optional ) = 0; virtual void Upload( CUploadJob * job, ITransferCallback * CallBack, HANDLE Token, QMErrInfo & ErrInfo ) = 0; virtual ~Downloader() {} protected: }; HRESULT CreateHttpDownloader( Downloader **ppDownloader, QMErrInfo *pErrInfo ); void DeleteHttpDownloader( Downloader * pDownloader ); extern DWORD g_dwDownloadThread; extern HWND ghPDWnd; // // conversion factor to go from time in milliseconds to time in 100-nanoseconds. // #define ONE_MSEC_IN_100_NSEC (10 * 1000) class CPeriodicTimer { public: CPeriodicTimer( LPSECURITY_ATTRIBUTES Security = NULL, BOOL ManualReset = FALSE, LPCTSTR Name = NULL ) { m_handle = CreateWaitableTimer( Security, ManualReset, Name ); if (!m_handle) { ThrowLastError(); } } ~CPeriodicTimer() { if (m_handle) { CloseHandle( m_handle ); } } BOOL Start( LONG Period, BOOL fResume = FALSE, PTIMERAPCROUTINE fn = NULL, LPVOID arg = NULL ) { LARGE_INTEGER Time; // // negative values are relative; positive values are absolute. // The period is in milliseconds, but the start time is in units of 100-nanoseconds, // Time.QuadPart = -1 * Period * ONE_MSEC_IN_100_NSEC; return SetWaitableTimer( m_handle, &Time, Period, fn, arg, fResume ); } BOOL Stop() { return CancelWaitableTimer( m_handle ); } BOOL Wait( LONG msec = INFINITE ) { if (WAIT_OBJECT_0 != WaitForSingleObject( m_handle, msec )) { return FALSE; } return TRUE; } private: HANDLE m_handle; }; // // Network rate is in bytes per second. // typedef float NETWORK_RATE; class CNetworkInterface { public: // // snapshot indices // enum { BLOCK_START = 0, BLOCK_END, BLOCK_INTERVAL_END, BLOCK_COUNT }; typedef float SECONDS; // // public interface // CNetworkInterface(); void Reset(); void SetInterfaceSpeed(); HRESULT TakeSnapshot( int SnapshotIndex ); HRESULT SetInterfaceIndex( const TCHAR host[] ); BOOL SetTimerInterval( SECONDS interval ); NETWORK_RATE GetInterfaceSpeed() { return m_ServerSpeed; } float GetPercentFree() { return m_PercentFree; } void ResetInterface() { m_InterfaceIndex = -1; } void Wait() { m_Timer.Wait(); } void StopTimer() { m_Timer.Stop(); } void CalculateIntervalAndBlockSize( UINT64 MaxBlockSize ); DWORD m_BlockSize; SECONDS m_BlockInterval; private: static const NETWORK_RATE DEFAULT_SPEED; struct NET_SNAPSHOT { LARGE_INTEGER TimeStamp; UINT BytesIn; UINT BytesOut; }; // // index of the interface that Wininet would use to talk to the server. // DWORD m_InterfaceIndex; // // the "start" and "end" pictures of network activity. // NET_SNAPSHOT m_Snapshots[BLOCK_COUNT]; // // the apparent speed of the connection to our server // NETWORK_RATE m_ServerSpeed; // // The local interface's apparent speed // NETWORK_RATE m_NetcardSpeed; // // // float m_PercentFree; // // Error value from previous snapshots in the current series {BLOCK_START, BLOCK_END, INTERVAL_END}. // HRESULT m_SnapshotError; // // true if all three snapshots in the current series are valid. // If so, then it is safe to calculate network speed and server speed. // bool m_SnapshotsValid; MIB_IFROW m_TempRow; // // The download thread sends only one packet per timer notification. // This is the timer. // CPeriodicTimer m_Timer; enum DOWNLOAD_STATE { DOWNLOADED_BLOCK = 0x55, SKIPPED_ONE_BLOCK, SKIPPED_TWO_BLOCKS }; enum DOWNLOAD_STATE m_state; static HRESULT FindInterfaceIndex( const TCHAR host[], DWORD * pIndex ); float GetTimeDifference( int start, int finish ); //throttle related DWORD BlockSizeFromInterval( SECONDS interval ); SECONDS IntervalFromBlockSize( DWORD BlockSize ); }; ///////////////////////////////////////////////////////////////////////////// // CProgressiveDL // class CProgressiveDL : public Downloader { public: CProgressiveDL( QMErrInfo *pErrInfo ); ~CProgressiveDL(); // pure virtual method from class Downloader virtual HRESULT Download( LPCTSTR szURL, LPCTSTR szDest, UINT64 Offset, ITransferCallback * CallBack, QMErrInfo *pErrInfo, HANDLE hToken, BOOL bThrottle, const PROXY_SETTINGS *pProxySettings, const CCredentialsContainer * Credentials, const StringHandle HostId = StringHandle() ); virtual HRESULT GetRemoteFileInformation( HANDLE hToken, LPCTSTR szURL, UINT64 * pFileSize, FILETIME *pFileTime, QMErrInfo *pErrInfo, const PROXY_SETTINGS * pProxySettings, const CCredentialsContainer * Credentials, const StringHandle HostId = StringHandle() ); virtual void Upload( CUploadJob * job, ITransferCallback * CallBack, HANDLE Token, QMErrInfo & ErrInfo ); // other methods private: //download related BOOL OpenLocalDownloadFile( LPCTSTR Path, UINT64 Offset, UINT64 Size, FILETIME UrlModificationTime, FILETIME * pFileTime ); BOOL CloseLocalFile(); BOOL WriteBlockToCache(LPBYTE lpBuffer, DWORD dwRead); BOOL SetFileTimes(); HRESULT DownloadFile( HANDLE hToken, const PROXY_SETTINGS * ProxySettings, const CCredentialsContainer * Credentials, LPCTSTR Url, LPCWSTR Path, UINT64 Offset, StringHandle HostId ); HRESULT GetNextBlock( ); BOOL DownloadBlock( void * Buffer, DWORD * pdwRead); HRESULT OpenConnection(); void CloseHandles(); BOOL IsAbortRequested() { return m_Callbacks->PollAbort(); } BOOL IsFileComplete() { if (m_CurrentOffset == m_wupdinfo->FileSize) { return TRUE; } return FALSE; } void ClearError(); void SetError( ERROR_SOURCE Source, ERROR_STYLE Style, UINT64 Code, char * comment = 0 ); BOOL IsErrorSet() { // If the file was aborted, the error wont // be set. if (QM_FILE_ABORTED == m_pQMInfo->result) { return TRUE; } if (QM_SERVER_FILE_CHANGED == m_pQMInfo->result ) { return TRUE; } if (m_pQMInfo->Style != 0) { return TRUE; } return FALSE; } BOOL GetRemoteResourceInformation( URL_INFO * Info, QMErrInfo * pQMErrInfo ); HRESULT SetRequestProxy( HINTERNET hRequest, URL_INFO & Info, const PROXY_SETTINGS * ProxySettings ); // // These are static so they they don't mess with member data accidentally // static bool DoesErrorIndicateNoISAPI( DWORD dwHttpError ); HRESULT CreateBlockUrl( LPTSTR lpszNewUrl, DWORD Length); HRESULT StartEncodedRangeRequest( DWORD Length ); HRESULT StartRangeRequest( DWORD Length ); //-------------------------------------------------------------------- HANDLE m_hFile; //download related URL_INFO * m_wupdinfo; UINT64 m_CurrentOffset; HINTERNET m_hOpenRequest; QMErrInfo *m_pQMInfo; ITransferCallback * m_Callbacks; BOOL m_bThrottle; HRESULT DownloadForegroundFile(); public: // // Tracks network statistics. // CNetworkInterface m_Network; }; extern CACHED_AUTOPROXY * g_ProxyCache; class CAbstractDataReader { public: virtual DWORD GetLength() const = 0; virtual HRESULT Rewind() = 0; virtual HRESULT Read( PVOID Buffer, DWORD Length, DWORD * pBytesRead ) = 0; virtual bool IsCancelled( DWORD BytesRead ) = 0; }; HRESULT SendRequest( HINTERNET hRequest, URL_INFO * Info, CAbstractDataReader * Reader = 0 ); HRESULT SetRequestCredentials( HINTERNET hRequest, const CCredentialsContainer & Container ); HRESULT SetRequestProxy( HINTERNET hRequest, PROXY_SETTINGS_CONTAINER * ProxySettings ); HRESULT OpenHttpRequest( LPCTSTR Verb, LPCTSTR Protocol, URL_INFO & Info, HINTERNET * phRequest ); URL_INFO * ConnectToUrl( LPCTSTR Url, const PROXY_SETTINGS * ProxySettings, const CCredentialsContainer * Credentials, LPCTSTR HostId, QMErrInfo * pErrInfo ); HRESULT GetRequestHeader( HINTERNET hRequest, DWORD HeaderIndex, LPCWSTR HeaderName, CAutoString & Destination, size_t MaxChars ); HRESULT GetResponseVersion( HINTERNET hRequest, unsigned * MajorVersion, unsigned * MinorVersion ); HRESULT AddRangeHeader( HINTERNET hRequest, UINT64 Start, UINT64 End ); HRESULT AddIf_Unmodified_SinceHeader( HINTERNET hRequest, const FILETIME &Time ); HRESULT CheckLanManHashDisabled ();