/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- * * The contents of this file are subject to the Netscape Public * License Version 1.1 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.mozilla.org/NPL/ * * Software distributed under the License is distributed on an "AS * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or * implied. See the License for the specific language governing * rights and limitations under the License. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is Netscape * Communications Corporation. Portions created by Netscape are * Copyright (C) 1998 Netscape Communications Corporation. All * Rights Reserved. * * Contributor(s): */ /* The TestProtocols tests the basic protocols architecture and can be used to test individual protocols as well. If this grows too big then we should split it to individual protocols. -Gagan Saksena 04/29/99 */ #include #ifdef WIN32 #include #endif #include "nspr.h" #include "nscore.h" #include "nsCOMPtr.h" #include "nsIEventQueueService.h" #include "nsIIOService.h" #include "nsIServiceManager.h" #include "nsIStreamListener.h" #include "nsIInputStream.h" #include "nsIBufferInputStream.h" #include "nsCRT.h" #include "nsIChannel.h" #include "nsIURL.h" #include "nsIHTTPChannel.h" #include "nsIHttpEventSink.h" #include "nsIInterfaceRequestor.h" #include "nsIDNSService.h" #include "nsISimpleEnumerator.h" #include "nsIHTTPHeader.h" #include "nsXPIDLString.h" // this test app handles cookies. #include "nsICookieService.h" static NS_DEFINE_CID(nsCookieServiceCID, NS_COOKIESERVICE_CID); static NS_DEFINE_CID(kEventQueueServiceCID, NS_EVENTQUEUESERVICE_CID); static NS_DEFINE_CID(kIOServiceCID, NS_IOSERVICE_CID); //static PRTime gElapsedTime; // enable when we time it... static int gKeepRunning = 0; static PRBool gVerbose = PR_FALSE; static nsIEventQueue* gEventQ = nsnull; static PRBool gTestAsyncOpen = PR_FALSE; class URLLoadInfo : public nsISupports { public: URLLoadInfo(const char* aUrl); virtual ~URLLoadInfo(); // ISupports interface... NS_DECL_ISUPPORTS const char* Name() { return mURLString.GetBuffer(); } PRInt32 mBytesRead; PRTime mTotalTime; PRTime mConnectTime; nsCString mURLString; }; URLLoadInfo::URLLoadInfo(const char *aUrl) : mURLString(aUrl) { NS_INIT_REFCNT(); mBytesRead = 0; mConnectTime = mTotalTime = PR_Now(); } URLLoadInfo::~URLLoadInfo() { } NS_IMPL_ISUPPORTS(URLLoadInfo,NS_GET_IID(nsISupports)); class TestHTTPEventSink : public nsIHTTPEventSink { public: TestHTTPEventSink(); virtual ~TestHTTPEventSink(); // ISupports interface... NS_DECL_ISUPPORTS // nsIHTTPEventSink interface... NS_IMETHOD OnAwaitingInput(nsISupports* i_Context); NS_IMETHOD OnHeadersAvailable(nsISupports* i_Context); NS_IMETHOD OnProgress(nsISupports* i_Context, PRUint32 i_Progress, PRUint32 i_ProgressMax); // OnRedirect gets fired only if you have set FollowRedirects on the handler! NS_IMETHOD OnRedirect(nsISupports* i_Context, nsIURI* i_NewLocation); }; TestHTTPEventSink::TestHTTPEventSink() { NS_INIT_REFCNT(); } TestHTTPEventSink::~TestHTTPEventSink() { } NS_IMPL_ISUPPORTS(TestHTTPEventSink,NS_GET_IID(nsIHTTPEventSink)); NS_IMETHODIMP TestHTTPEventSink::OnAwaitingInput(nsISupports* context) { printf("\n+++ TestHTTPEventSink::OnAwaitingInput +++\n"); return NS_OK; } NS_IMETHODIMP TestHTTPEventSink::OnHeadersAvailable(nsISupports* context) { nsCOMPtr enumerator; nsCOMPtr pHTTPCon(do_QueryInterface(context)); PRBool bMoreHeaders; if (pHTTPCon) { nsXPIDLCString value; printf("Channel Info:\n"); pHTTPCon->GetContentType(getter_Copies(value)); printf("\tContent-Type: %s\n", (const char*)value); pHTTPCon->GetCharset(getter_Copies(value)); printf("\tCharset: %s\n", (const char*)value); pHTTPCon->GetRequestHeaderEnumerator(getter_AddRefs(enumerator)); printf("Request headers:\n"); enumerator->HasMoreElements(&bMoreHeaders); while (bMoreHeaders) { nsCOMPtr item; nsCOMPtr header; enumerator->GetNext(getter_AddRefs(item)); header = do_QueryInterface(item); if (header) { nsCOMPtr key; nsAutoString field; header->GetField(getter_AddRefs(key)); key->ToString(field); nsCAutoString theField(field); printf("\t%s: ", theField.GetBuffer()); header->GetValue(getter_Copies(value)); printf("%s\n", (const char*)value); } enumerator->HasMoreElements(&bMoreHeaders); } pHTTPCon->GetResponseHeaderEnumerator(getter_AddRefs(enumerator)); printf("Response headers:\n"); enumerator->HasMoreElements(&bMoreHeaders); while (bMoreHeaders) { nsCOMPtr item; nsCOMPtr header; enumerator->GetNext(getter_AddRefs(item)); header = do_QueryInterface(item); if (header) { nsCOMPtr key; nsAutoString field; header->GetField(getter_AddRefs(key)); key->ToString(field); nsCAutoString theField(field); printf("\t%s: ", theField.GetBuffer()); header->GetValue(getter_Copies(value)); printf("%s\n", (const char*)value); } enumerator->HasMoreElements(&bMoreHeaders); } } if (gVerbose) { printf("\n+++ TestHTTPEventSink::OnHeadersAvailable +++\n"); nsCOMPtr pHTTPCon(do_QueryInterface(context)); if (pHTTPCon) { char* type; //optimize later TODO allow atoms here...! intead of just the header strings pHTTPCon->GetContentType(&type); if (type) { printf("\nReceiving ... %s\n", type); nsCRT::free(type); } } } return NS_OK; } NS_IMETHODIMP TestHTTPEventSink::OnProgress(nsISupports* context, PRUint32 i_Progress, PRUint32 i_ProgressMax) { printf("\n+++ TestHTTPEventSink::OnProgress +++\n"); return NS_OK; } NS_IMETHODIMP TestHTTPEventSink::OnRedirect(nsISupports* context, nsIURI* i_NewLocation) { printf("\n+++ TestHTTPEventSink::OnRedirect +++\n"); return NS_OK; } //////////////////////////////////////////////////////////////////////////////// class InputTestConsumer : public nsIStreamListener { public: InputTestConsumer(); virtual ~InputTestConsumer(); NS_DECL_ISUPPORTS NS_DECL_NSISTREAMOBSERVER NS_DECL_NSISTREAMLISTENER void SetAsyncOpenCompleted() { mAsyncOpenCompleted = PR_TRUE; } protected: PRBool mAsyncOpenCompleted; }; //////////////////////////////////////////////////////////////////////////////// class OpenObserver : public nsIStreamObserver { public: NS_DECL_ISUPPORTS NS_DECL_NSISTREAMOBSERVER OpenObserver(InputTestConsumer* cons) : mInputConsumer(cons) { NS_INIT_REFCNT(); } virtual ~OpenObserver() {} protected: InputTestConsumer* mInputConsumer; }; NS_IMPL_ISUPPORTS1(OpenObserver, nsIStreamObserver); NS_IMETHODIMP OpenObserver::OnStartRequest(nsIChannel* channel, nsISupports* context) { printf("\n+++ OpenObserver::OnStartRequest +++. Context = %p\n", context); char* type; PRInt32 length = -1; nsresult rv; rv = channel->GetContentType(&type); NS_ASSERTION(NS_SUCCEEDED(rv), "GetContentType failed"); rv = channel->GetContentLength(&length); NS_ASSERTION(NS_SUCCEEDED(rv), "GetContentLength failed"); printf(" contentType = %s length = %d\n", type, length); nsCRT::free(type); mInputConsumer->SetAsyncOpenCompleted(); return NS_OK; } NS_IMETHODIMP OpenObserver::OnStopRequest(nsIChannel* channel, nsISupports* context, nsresult aStatus, const PRUnichar* aMsg) { printf("\n+++ OpenObserver::OnStopRequest (status = %x) +++." "\tContext = %p\n", aStatus, context); return NS_OK; } //////////////////////////////////////////////////////////////////////////////// InputTestConsumer::InputTestConsumer() : mAsyncOpenCompleted(PR_FALSE) { NS_INIT_REFCNT(); } InputTestConsumer::~InputTestConsumer() { } NS_IMPL_ISUPPORTS(InputTestConsumer,NS_GET_IID(nsIStreamListener)); NS_IMETHODIMP InputTestConsumer::OnStartRequest(nsIChannel* channel, nsISupports* context) { NS_ASSERTION(!gTestAsyncOpen || mAsyncOpenCompleted, "AsyncOpen failed"); URLLoadInfo* info = (URLLoadInfo*)context; if (info) { info->mConnectTime = PR_Now() - info->mConnectTime; } if (gVerbose) { printf("\nStarted loading: %s\n", info ? info->Name() : "UNKNOWN URL"); } /* nsCOMPtr pURI(do_QueryInterface(context)); char* location = nsnull; if (pURI) { pURI->GetSpec(&location); } printf("\nStarted loading: %s\n", location ? location : "UNKNOWN URL"); if (location) { nsCRT::free(location); } */ return NS_OK; } NS_IMETHODIMP InputTestConsumer::OnDataAvailable(nsIChannel* channel, nsISupports* context, nsIInputStream *aIStream, PRUint32 aSourceOffset, PRUint32 aLength) { NS_ASSERTION(!gTestAsyncOpen || mAsyncOpenCompleted, "AsyncOpen failed"); char buf[1025]; PRUint32 amt, size; nsresult rv; URLLoadInfo* info = (URLLoadInfo*)context; while (aLength) { size = PR_MIN(aLength, sizeof(buf)); rv = aIStream->Read(buf, size, &amt); if (NS_FAILED(rv)) { NS_ASSERTION((NS_BASE_STREAM_WOULD_BLOCK != rv), "The stream should never block."); return rv; } if (gVerbose) { buf[amt] = '\0'; puts(buf); } if (info) { info->mBytesRead += amt; } aLength -= amt; } return NS_OK; } NS_IMETHODIMP InputTestConsumer::OnStopRequest(nsIChannel* channel, nsISupports* context, nsresult aStatus, const PRUnichar* aMsg) { NS_ASSERTION(!gTestAsyncOpen || mAsyncOpenCompleted, "AsyncOpen failed"); URLLoadInfo* info = (URLLoadInfo*)context; if (info) { double connectTime; double readTime; PRUint32 httpStatus; PRBool bHTTPURL = PR_FALSE; info->mTotalTime = PR_Now() - info->mTotalTime; connectTime = (info->mConnectTime/1000.0)/1000.0; readTime = ((info->mTotalTime-info->mConnectTime)/1000.0)/1000.0; nsCOMPtr pHTTPCon(do_QueryInterface(channel)); if (pHTTPCon) { pHTTPCon->GetResponseStatus(&httpStatus); bHTTPURL = PR_TRUE; } printf("\nFinished loading: %s Status Code: %x\n", info->Name(), aStatus); if (bHTTPURL) printf("\tHTTP Status: %u\n", httpStatus); if (NS_ERROR_UNKNOWN_HOST == aStatus) { printf("\tDNS lookup failed.\n"); } printf("\tRead: %d bytes.\n", info->mBytesRead); printf("\tTime to connect: %.3f seconds\n", connectTime); printf("\tTime to read: %.3f seconds.\n", readTime); if (readTime > 0.0) { printf("\tThroughput: %.0f bps.\n", (info->mBytesRead*8)/readTime); } else { printf("\tThroughput: REAL FAST!!\n"); } } else { printf("\nFinished loading: UNKNOWN URL. Status Code: %x\n", aStatus); } /* nsCOMPtr pURI(do_QueryInterface(context)); char* location = nsnull; if (pURI) { pURI->GetSpec(&location); } printf("\nFinished loading: %s Status Code: %x\n", location ? location : "UNKNOWN URL", aStatus); if (location) { nsCRT::free(location); } */ gKeepRunning -= 1; return NS_OK; } //////////////////////////////////////////////////////////////////////////////// class nsNotificationCallbacks : public nsIInterfaceRequestor { public: NS_DECL_ISUPPORTS nsNotificationCallbacks() { NS_INIT_REFCNT(); } NS_IMETHOD GetInterface(const nsIID& eventSinkIID, void* *result) { nsresult rv = NS_ERROR_FAILURE; if (eventSinkIID.Equals(NS_GET_IID(nsIHTTPEventSink))) { TestHTTPEventSink *sink; sink = new TestHTTPEventSink(); if (sink == nsnull) return NS_ERROR_OUT_OF_MEMORY; NS_ADDREF(sink); rv = sink->QueryInterface(eventSinkIID, result); NS_RELEASE(sink); } return rv; } }; NS_IMPL_ISUPPORTS1(nsNotificationCallbacks, nsIInterfaceRequestor) //////////////////////////////////////////////////////////////////////////////// nsresult StartLoadingURL(const char* aUrlString) { nsresult rv; NS_WITH_SERVICE(nsIIOService, pService, kIOServiceCID, &rv); if (pService) { nsCOMPtr pURL; rv = pService->NewURI(aUrlString, nsnull, getter_AddRefs(pURL)); if (NS_FAILED(rv)) { printf("ERROR: NewURI failed for %s\n", aUrlString); return rv; } nsCOMPtr pChannel; nsNotificationCallbacks* callbacks = new nsNotificationCallbacks(); if (!callbacks) { NS_ERROR("Failed to create a new consumer!"); return NS_ERROR_OUT_OF_MEMORY;; } NS_ADDREF(callbacks); // Async reading thru the calls of the event sink interface rv = pService->NewChannelFromURI("load", pURL, nsnull, // loadGroup callbacks, // notificationCallbacks nsIChannel::LOAD_NORMAL, nsnull, // originalURI getter_AddRefs(pChannel)); NS_RELEASE(callbacks); if (NS_FAILED(rv)) { printf("ERROR: NewChannelFromURI failed for %s\n", aUrlString); return rv; } /* You may optionally add/set other headers on this request object. This is done by QI for the specific protocolConnection. */ nsCOMPtr pHTTPCon(do_QueryInterface(pChannel)); if (pHTTPCon) { // Setting a sample header. nsCOMPtr sampleHeader; sampleHeader = NS_NewAtom("sample-header-minus-the-colon"); rv = pHTTPCon->SetRequestHeader(sampleHeader, "Sample-Value"); if (NS_FAILED(rv)) return rv; } InputTestConsumer* listener; listener = new InputTestConsumer; NS_IF_ADDREF(listener); if (!listener) { NS_ERROR("Failed to create a new stream listener!"); return NS_ERROR_OUT_OF_MEMORY;; } if (gTestAsyncOpen) { OpenObserver* obs = new OpenObserver(listener); if (obs == nsnull) return NS_ERROR_OUT_OF_MEMORY; NS_ADDREF(obs); rv = pChannel->AsyncOpen(obs, nsnull); NS_RELEASE(obs); if (NS_FAILED(rv)) { NS_ERROR("Error: AsyncOpen failed..."); return rv; } } URLLoadInfo* info; info = new URLLoadInfo(aUrlString); NS_IF_ADDREF(info); if (!info) { NS_ERROR("Failed to create a load info!"); return NS_ERROR_OUT_OF_MEMORY; } rv = pChannel->AsyncRead(0, // staring position -1, // number of bytes to read info, // ISupports context listener); // IStreamListener consumer if (NS_SUCCEEDED(rv)) { gKeepRunning += 1; } NS_RELEASE(listener); NS_RELEASE(info); } return rv; } nsresult LoadURLsFromFile(char *aFileName) { nsresult rv = NS_OK; PRInt32 len, offset; PRFileDesc* fd; char buffer[1024]; nsCString fileBuffer; nsCAutoString urlString; fd = PR_Open(aFileName, PR_RDONLY, 777); if (!fd) { return NS_ERROR_FAILURE; } // Keep reading the file until EOF (or an error) is reached... do { len = PR_Read(fd, buffer, sizeof(buffer)); if (len>0) { fileBuffer.Append(buffer, len); // Treat each line as a URL... while ((offset = fileBuffer.FindChar('\n')) != -1) { fileBuffer.Left(urlString, offset); fileBuffer.Cut(0, offset+1); urlString.StripChars("\r"); if (urlString.Length()) { printf("\t%s\n", urlString.GetBuffer()); rv = StartLoadingURL(urlString.GetBuffer()); } } } } while (len>0); // If anything is left in the fileBuffer, treat it as a URL... fileBuffer.StripChars("\r"); if (fileBuffer.Length()) { printf("\t%s\n", fileBuffer.GetBuffer()); StartLoadingURL(fileBuffer.GetBuffer()); } PR_Close(fd); return NS_OK; } nsresult NS_AutoregisterComponents() { nsresult rv = nsComponentManager::AutoRegister(nsIComponentManager::NS_Startup, NULL /* default */); return rv; } int main(int argc, char* argv[]) { nsresult rv= (nsresult)-1; if (argc < 2) { printf("usage: %s [-verbose] [-file ] ... \n", argv[0]); return -1; } /* The following code only deals with XPCOM registration stuff. and setting up the event queues. Copied from TestSocketIO.cpp */ rv = NS_AutoregisterComponents(); if (NS_FAILED(rv)) return rv; // Create the Event Queue for this thread... NS_WITH_SERVICE(nsIEventQueueService, eventQService, kEventQueueServiceCID, &rv); if (NS_FAILED(rv)) return rv; rv = eventQService->CreateThreadEventQueue(); if (NS_FAILED(rv)) return rv; eventQService->GetThreadEventQueue(PR_CurrentThread(), &gEventQ); #if 0 // Jud sez // fire up an instance of the cookie manager. // I'm doing this using the serviceManager for convenience's sake. // Presumably an application will init it's own cookie service a // different way (this way works too though). NS_WITH_SERVICE(nsICookieService, cookieService, nsCookieServiceCID, &rv); if (NS_FAILED(rv)) return rv; #endif // NECKO int i; printf("\nTrying to load:\n"); for (i=1; iGetEvent(&gEvent); rv = gEventQ->HandleEvent(gEvent); #endif /* XP_UNIX */ #endif /* !WIN32 */ } return rv; }