// ClientDlg.cpp : implementation file // #include "stdafx.h" #include "Client.h" #include "ClientDlg.h" #include "afxdialogex.h" // CClientDlg dialog #define DEFAULT_PATH _T("req/path?p1=v1&p2=v2") #define DEFAULT_ADDRESS _T("127.0.0.1") #define DEFAULT_PORT _T("8080") CClientDlg::CClientDlg(CWnd* pParent /*=NULL*/) : CDialogEx(CClientDlg::IDD, pParent) { m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME); } void CClientDlg::DoDataExchange(CDataExchange* pDX) { CDialogEx::DoDataExchange(pDX); DDX_Control(pDX, IDC_SEND, m_Send); DDX_Control(pDX, IDC_INFO, m_Info); DDX_Control(pDX, IDC_ADDRESS, m_Address); DDX_Control(pDX, IDC_PORT, m_Port); DDX_Control(pDX, IDC_ASYNC, m_Async); DDX_Control(pDX, IDC_START, m_Start); DDX_Control(pDX, IDC_STOP, m_Stop); DDX_Control(pDX, IDC_METHOD, m_Method); DDX_Control(pDX, IDC_SCHEMA, m_Schema); DDX_Control(pDX, IDC_PATH, m_Path); DDX_Control(pDX, IDC_HEADER_NAME, m_HeaderName); DDX_Control(pDX, IDC_HEADER_VALUE, m_HeaderValue); DDX_Control(pDX, IDC_HEADER_ADD, m_HeaderAdd); DDX_Control(pDX, IDC_HEADERS, m_Headers); DDX_Control(pDX, IDC_BODY, m_Body); } BEGIN_MESSAGE_MAP(CClientDlg, CDialogEx) ON_WM_PAINT() ON_WM_QUERYDRAGICON() ON_BN_CLICKED(IDC_SEND, &CClientDlg::OnBnClickedSend) ON_BN_CLICKED(IDC_START, &CClientDlg::OnBnClickedStart) ON_BN_CLICKED(IDC_STOP, &CClientDlg::OnBnClickedStop) ON_BN_CLICKED(IDC_HEADER_ADD, &CClientDlg::OnBnClickedHeaderAdd) ON_CBN_SELCHANGE(IDC_METHOD, &CClientDlg::OnCbnSelchangeMethod) ON_MESSAGE(USER_INFO_MSG, OnUserInfoMsg) ON_WM_VKEYTOITEM() END_MESSAGE_MAP() // CClientDlg message handlers BOOL CClientDlg::OnInitDialog() { CDialogEx::OnInitDialog(); // Set the icon for this dialog. The framework does this automatically // when the application's main window is not a dialog SetIcon(m_hIcon, TRUE); // Set big icon SetIcon(m_hIcon, FALSE); // Set small icon // TODO: Add extra initialization here m_Method.SetCurSel(3); m_Schema.SetCurSel(0); m_Path.SetWindowText(DEFAULT_PATH); m_Address.SetWindowText(DEFAULT_ADDRESS); m_Port.SetWindowText(DEFAULT_PORT); m_Async.SetCheck(BST_CHECKED); ::SetMainWnd(this); ::SetInfoList(&m_Info); SetAppState(ST_STOPPED); m_bAsyncConn = FALSE; return TRUE; // return TRUE unless you set the focus to a control } // If you add a minimize button to your dialog, you will need the code below // to draw the icon. For MFC applications using the document/view model, // this is automatically done for you by the framework. void CClientDlg::OnPaint() { if (IsIconic()) { CPaintDC dc(this); // device context for painting SendMessage(WM_ICONERASEBKGND, reinterpret_cast(dc.GetSafeHdc()), 0); // Center icon in client rectangle int cxIcon = GetSystemMetrics(SM_CXICON); int cyIcon = GetSystemMetrics(SM_CYICON); CRect rect; GetClientRect(&rect); int x = (rect.Width() - cxIcon + 1) / 2; int y = (rect.Height() - cyIcon + 1) / 2; // Draw the icon dc.DrawIcon(x, y, m_hIcon); } else { CDialogEx::OnPaint(); } } // The system calls this function to obtain the cursor to display while the user drags // the minimized window. HCURSOR CClientDlg::OnQueryDragIcon() { return static_cast(m_hIcon); } BOOL CClientDlg::PreTranslateMessage(MSG* pMsg) { if ( pMsg->message == WM_KEYDOWN &&( pMsg->wParam == VK_ESCAPE || pMsg->wParam == VK_CANCEL || (pMsg->wParam == VK_RETURN && pMsg->hwnd == this->GetSafeHwnd()) )) return TRUE; return CDialog::PreTranslateMessage(pMsg); } void CClientDlg::SetAppState(EnAppState state) { m_enState = state; if(this->GetSafeHwnd() == nullptr) return; m_Async.EnableWindow(m_enState == ST_STOPPED); m_Start.EnableWindow(m_enState == ST_STOPPED); m_Stop.EnableWindow(m_enState == ST_STARTED); m_Send.EnableWindow(m_enState == ST_STARTED); m_Schema.EnableWindow(m_enState == ST_STOPPED); m_Address.EnableWindow(m_enState == ST_STOPPED); m_Port.EnableWindow(m_enState == ST_STOPPED); m_Body.EnableWindow(m_Method.GetCurSel() <= 2); m_Path.EnableWindow(m_Method.GetCurSel() != 8); } void CClientDlg::OnBnClickedHeaderAdd() { CString strName; CString strValue; m_HeaderName.GetWindowText(strName); m_HeaderValue.GetWindowText(strValue); strName.Trim(); strValue.Trim(); if(!strName.IsEmpty()) { if(strName == _T("Content-Length")) { MessageBox(_T("'Content-Length' header can't be specified!"), _T("Add Header"), MB_OK | MB_ICONINFORMATION); m_HeaderName.SetFocus(); return; } CString strHeader; strHeader.Format(_T("%s: %s"), strName, strValue); m_Headers.AddString(strHeader); m_HeaderName.SetWindowText(_T("")); m_HeaderValue.SetWindowText(_T("")); } } void CClientDlg::OnBnClickedSend() { USES_CONVERSION; CString strMethod; CString strSchema; CString strAddress; CString strPort; CString strPath; m_Method.GetWindowText(strMethod); m_Schema.GetWindowText(strSchema); m_Address.GetWindowText(strAddress); m_Port.GetWindowText(strPort); if(m_Method.GetCurSel() != 8) { m_Path.GetWindowText(strPath); strPath.Trim(); if(strPath.IsEmpty() || strPath.GetAt(0) != '/') strPath.Insert(0, '/'); } THeaderMap headers; int iCount = m_Headers.GetCount(); for(int i = 0; i < iCount; i++) { CString strHeader; m_Headers.GetText(i, strHeader); int j = 0; CString strName = strHeader.Tokenize(_T(": "), j); CString strValue = strHeader.Mid(j + 1); headers.emplace(THeaderMap::value_type(T2A(strName), T2A(strValue))); } THeaderMapCI it = headers.find("Host"); if(it == headers.end() || it->second.IsEmpty()) { CString strHost; strHost.Format(_T("%s:%s"), strAddress, strPort); headers.erase("Host"); headers.emplace(THeaderMap::value_type("Host", T2A(strHost))); } CStringA strBodyA; CStringA strPathA; if(m_Method.GetCurSel() <= 2) { CString strBody; m_Body.GetWindowText(strBody); strBodyA = T2A(strBody); } else if(m_Method.GetCurSel() == 8) strPathA = headers.find("Host")->second; if(strPathA.IsEmpty()) strPathA = T2A(strPath); DWORD dwIndex = 0; DWORD dwSize = (DWORD)headers.size(); unique_ptr szHeaders(new THeader[dwSize]); for(THeaderMapCI it = headers.begin(), end = headers.end(); it != end; ++it, ++dwIndex) { szHeaders[dwIndex].name = it->first; szHeaders[dwIndex].value = it->second; } CString strContent; strContent.Format(_T("[%s] %s://%s:%s%s"), strMethod, strSchema, strAddress, strPort, strPath); if(m_pClient->SendRequest(T2A(strMethod), strPathA, szHeaders.get(), dwSize, (const BYTE*)(LPCSTR)strBodyA, strBodyA.GetLength())) ::LogSend(m_pClient->GetConnectionID(), strContent); else ::LogSendFail(m_pClient->GetConnectionID(), ::GetLastError(), ::GetSocketErrorDesc(SE_DATA_SEND)); } void CClientDlg::OnBnClickedStart() { SetAppState(ST_STARTING); g_SSL.Cleanup(); if(!g_SSL.Initialize(SSL_SM_CLIENT, SSL_VM_NONE, g_c_lpszPemCertFile, g_c_lpszPemKeyFile, g_c_lpszKeyPasswod, g_c_lpszCAPemCertFileOrPath)) { ::LogClientStartFail(::GetLastError(), _T("initialize SSL env fail")); SetAppState(ST_STOPPED); return; } CString strAddress; CString strPort; m_Address.GetWindowText(strAddress); m_Port.GetWindowText(strPort); USHORT usPort = (USHORT)_ttoi(strPort); m_bAsyncConn = m_Async.GetCheck(); BOOL isHttp = m_Schema.GetCurSel() == 0; if(isHttp) m_pClient.reset((IHttpClient*)(new CHttpClient(this))); else m_pClient.reset((IHttpClient*)(new CHttpsClient(this))); ::LogClientStarting(strAddress, usPort); if(m_pClient->Start(strAddress, usPort, m_bAsyncConn)) { } else { ::LogClientStartFail(m_pClient->GetLastError(), m_pClient->GetLastErrorDesc()); SetAppState(ST_STOPPED); } } void CClientDlg::OnBnClickedStop() { SetAppState(ST_STOPPING); if(m_pClient->Stop()) ::LogClientStopping(m_pClient->GetConnectionID()); else ASSERT(FALSE); } void CClientDlg::OnCbnSelchangeMethod() { m_Body.EnableWindow(m_Method.GetCurSel() <= 2); m_Path.EnableWindow(m_Method.GetCurSel() != 8); } int CClientDlg::OnVKeyToItem(UINT nKey, CListBox* pListBox, UINT nIndex) { if(nKey == 'C') pListBox->ResetContent(); else if(nKey == 'D' && pListBox == &m_Headers) { int index = pListBox->GetCurSel(); if(index != LB_ERR) { pListBox->DeleteString(index); int count = pListBox->GetCount(); if(index < count) pListBox->SetCurSel(index); else if(index > 0) pListBox->SetCurSel(index - 1); } } return __super::OnVKeyToItem(nKey, pListBox, nIndex); } LRESULT CClientDlg::OnUserInfoMsg(WPARAM wp, LPARAM lp) { info_msg* msg = (info_msg*)wp; ::LogInfoMsg(msg); return 0; } EnHandleResult CClientDlg::OnConnect(IClient* pClient) { TCHAR szAddress[40]; int iAddressLen = sizeof(szAddress) / sizeof(TCHAR); USHORT usPort; pClient->GetLocalAddress(szAddress, iAddressLen, usPort); ::PostOnConnect(pClient->GetConnectionID(), szAddress, usPort); return HR_OK; } EnHandleResult CClientDlg::OnHandShake(IClient* pClient) { ::PostOnHandShake(pClient->GetConnectionID()); SetAppState(ST_STARTED); return HR_OK; } EnHandleResult CClientDlg::OnSend(IClient* pClient, const BYTE* pData, int iLength) { ::PostOnSend(pClient->GetConnectionID(), pData, iLength); return HR_OK; } EnHandleResult CClientDlg::OnReceive(IClient* pClient, const BYTE* pData, int iLength) { ::PostOnReceive(pClient->GetConnectionID(), pData, iLength); return HR_OK; } EnHandleResult CClientDlg::OnClose(IClient* pClient, EnSocketOperation enOperation, int iErrorCode) { iErrorCode == SE_OK ? ::PostOnClose(pClient->GetConnectionID()) : ::PostOnError(pClient->GetConnectionID(), enOperation, iErrorCode) ; SetAppState(ST_STOPPED); return HR_OK; } // ------------------------------------------------------------------------------------------------------------- // EnHttpParseResult CClientDlg::OnMessageBegin(IHttp* pHttp) { ::PostOnMessageBegin(((IHttpClient*)pHttp)->GetConnectionID()); return HPR_OK; } EnHttpParseResult CClientDlg::OnStatusLine(IHttp* pHttp, USHORT usStatusCode, LPCSTR lpszDesc) { ::PostOnStatusLine(((IHttpClient*)pHttp)->GetConnectionID(), usStatusCode, lpszDesc); return HPR_OK; } EnHttpParseResult CClientDlg::OnHeader(IHttp* pHttp, LPCSTR lpszName, LPCSTR lpszValue) { ::PostOnHeader(((IHttpClient*)pHttp)->GetConnectionID(), lpszName, lpszValue); return HPR_OK; } EnHttpParseResult CClientDlg::OnHeadersComplete(IHttp* pHttp) { IHttpClient* pHttpClient = (IHttpClient*)pHttp; CheckSetCookie(pHttpClient); CStringA strSummary = GetHeaderSummary(pHttpClient, " ", 0, TRUE); ::PostOnHeadersComplete(pHttpClient->GetConnectionID(), strSummary); return HPR_OK; } EnHttpParseResult CClientDlg::OnBody(IHttp* pHttp, const BYTE* pData, int iLength) { ::PostOnBody(((IHttpClient*)pHttp)->GetConnectionID(), pData, iLength); return HPR_OK; } EnHttpParseResult CClientDlg::OnChunkHeader(IHttp* pHttp, int iLength) { ::PostOnChunkHeader(((IHttpClient*)pHttp)->GetConnectionID(), iLength); return HPR_OK; } EnHttpParseResult CClientDlg::OnChunkComplete(IHttp* pHttp) { ::PostOnChunkComplete(((IHttpClient*)pHttp)->GetConnectionID()); return HPR_OK; } EnHttpParseResult CClientDlg::OnMessageComplete(IHttp* pHttp) { ::PostOnMessageComplete(((IHttpClient*)pHttp)->GetConnectionID()); return HPR_OK; } EnHttpParseResult CClientDlg::OnUpgrade(IHttp* pHttp, EnHttpUpgradeType enUpgradeType) { ::PostOnUpgrade(((IHttpClient*)pHttp)->GetConnectionID(), enUpgradeType); return HPR_OK; } EnHttpParseResult CClientDlg::OnParseError(IHttp* pHttp, int iErrorCode, LPCSTR lpszErrorDesc) { ::PostOnParseError(((IHttpClient*)pHttp)->GetConnectionID(), iErrorCode, lpszErrorDesc); return HPR_OK; } void CClientDlg::CheckSetCookie(IHttpClient* pHttpClient) { DWORD dwHeaderCount = 0; pHttpClient->GetHeaders("Set-Cookie", nullptr, dwHeaderCount); if(dwHeaderCount == 0) return; unique_ptr values(new LPCSTR[dwHeaderCount]); VERIFY(pHttpClient->GetHeaders("Set-Cookie", values.get(), dwHeaderCount)); for(DWORD i = 0; i < dwHeaderCount; i++) { CStringA strValue = values[i]; int j = 0; CStringA strItem = strValue.Tokenize("; ", j); if(j <= 0) continue; int k = strItem.Find('='); if(k <= 0) continue; pHttpClient->AddCookie(strItem.Left(k), strItem.Mid(k + 1)); } } CStringA CClientDlg::GetHeaderSummary(IHttpClient* pHttpClient, LPCSTR lpszSep, int iSepCount, BOOL bWithContentLength) { CStringA SEP1; for(int i = 0; i < iSepCount; i++) SEP1 += lpszSep; CStringA SEP2(SEP1); SEP2 += lpszSep; CStringA strResult; strResult.AppendFormat("%s[Status Fields]%s", SEP1, CRLF); strResult.AppendFormat("%s%13s: %u%s", SEP2, "Status Code", pHttpClient->GetStatusCode(), CRLF); DWORD dwHeaderCount = 0; pHttpClient->GetAllHeaders(nullptr, dwHeaderCount); strResult.AppendFormat("%s[Response Headers]%s", SEP1, CRLF); if(dwHeaderCount == 0) strResult.AppendFormat("%s(no header)%s", SEP2, CRLF); else { unique_ptr headers(new THeader[dwHeaderCount]); VERIFY(pHttpClient->GetAllHeaders(headers.get(), dwHeaderCount)); for(DWORD i = 0; i < dwHeaderCount; i++) strResult.AppendFormat("%s%s: %s%s", SEP2, headers[i].name, headers[i].value, CRLF); } DWORD dwCookieCount = 0; pHttpClient->GetAllCookies(nullptr, dwCookieCount); strResult.AppendFormat("%s[Cookies]%s", SEP1, CRLF); if(dwCookieCount == 0) strResult.AppendFormat("%s(no cookie)%s", SEP2, CRLF); else { unique_ptr cookies(new TCookie[dwCookieCount]); VERIFY(pHttpClient->GetAllCookies(cookies.get(), dwCookieCount)); for(DWORD i = 0; i < dwCookieCount; i++) strResult.AppendFormat("%s%s: %s%s", SEP2, cookies[i].name, cookies[i].value, CRLF); } CStringA strVersion; ::GetHttpVersionStr((EnHttpVersion)pHttpClient->GetVersion(), strVersion); EnHttpUpgradeType enUpgType = pHttpClient->GetUpgradeType(); LPCSTR lpszUpgrade = enUpgType != HUT_NONE ? "true" : "false"; LPCSTR lpszKeepAlive = pHttpClient->IsKeepAlive() ? "true" : "false"; strResult.AppendFormat("%s[Basic Info]%s", SEP1, CRLF); strResult.AppendFormat("%s%13s: %s%s", SEP2, "Version", strVersion, CRLF); strResult.AppendFormat("%s%13s: %u%s", SEP2, "Status Code", pHttpClient->GetStatusCode(), CRLF); strResult.AppendFormat("%s%13s: %s%s", SEP2, "IsUpgrade", lpszUpgrade, CRLF); if(enUpgType != HUT_NONE) strResult.AppendFormat("%s%13s: %d%s", SEP2, "UpgradeType", enUpgType, CRLF); strResult.AppendFormat("%s%13s: %s%s", SEP2, "IsKeepAlive", lpszKeepAlive, CRLF); if(bWithContentLength) strResult.AppendFormat("%s%13s: %lld%s", SEP2, "ContentLength", pHttpClient->GetContentLength(), CRLF); strResult.AppendFormat("%s%13s: %s%s", SEP2, "ContentType", pHttpClient->GetContentType(), CRLF); return strResult; }