Files correlati : Ricompilazione Demo : [ ] Commento : Riportata la versione P@rtners 2.0 patch 349 git-svn-id: svn://10.65.10.50/trunk@10573 c028cbd2-c16b-5b4b-a496-9718f37d4682
		
			
				
	
	
		
			587 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			587 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
		
			Executable File
		
	
	
	
	
| /////////////////////////////////////////////////////////////////////////////
 | |
| // Name:        BaseServ.cpp
 | |
| // Purpose:     Simple base Server
 | |
| // Author:      Guy
 | |
| // Modified by:
 | |
| // Created:     21/08/2002
 | |
| // Copyright:   (c) 2002 Guy
 | |
| // Licence:     wxWindows licence
 | |
| /////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| // ==========================================================================
 | |
| // declarations
 | |
| // ==========================================================================
 | |
| 
 | |
| // --------------------------------------------------------------------------
 | |
| // headers
 | |
| // --------------------------------------------------------------------------
 | |
| 
 | |
| #include "BaseServ.h"
 | |
| 
 | |
| #include <wx/config.h>
 | |
| #include <wx/mimetype.h>
 | |
| #include <wx/sckstrm.h>
 | |
| #include <wx/msw/iniconf.h> 
 | |
| 
 | |
| #include <wx/app.h>
 | |
| 
 | |
| ///////////////////////////////////////////////////////////
 | |
| // Utilities
 | |
| ///////////////////////////////////////////////////////////
 | |
| 
 | |
| wxString Date2String(const wxDateTime& date)
 | |
| {
 | |
| 	return date.Format("%d-%m-%Y");
 | |
| }
 | |
| 
 | |
| wxDateTime String2Date(const wxString& str)
 | |
| {
 | |
| 	int d, m, y;
 | |
| 	sscanf(str, "%d-%d-%d", &d, &m, &y);
 | |
| 	wxDateTime date(d, wxDateTime::Month(m-1), y);
 | |
| 	return date;
 | |
| }
 | |
| 
 | |
| // --------------------------------------------------------------------------
 | |
| // classes
 | |
| // --------------------------------------------------------------------------
 | |
| 
 | |
| ///////////////////////////////////////////////////////////
 | |
| // THashTable
 | |
| ///////////////////////////////////////////////////////////
 | |
| 
 | |
| void THashTable::Put(const wxString& key, const wxString& value)
 | |
| {
 | |
| 	THashString* entry = new THashString(value);
 | |
| 	m_Hash.Put(key, entry);
 | |
| }
 | |
| 
 | |
| wxString THashTable::Get(const wxChar* key) const
 | |
| {
 | |
| 	wxString strValue;
 | |
| 	THashString* str = (THashString*)m_Hash.Get(key);
 | |
| 	if (str != NULL)
 | |
| 		strValue = str->m_str;
 | |
| 	return strValue;
 | |
| }
 | |
| 
 | |
| THashTable::THashTable(size_t size) : m_Hash(wxKEY_STRING, size) 
 | |
| { 
 | |
| 	m_Hash.DeleteContents(true); 
 | |
| }
 | |
| 
 | |
| ///////////////////////////////////////////////////////////
 | |
| // TBaseServerApp
 | |
| ///////////////////////////////////////////////////////////
 | |
| 
 | |
| TBaseServerApp& GetServerApp()
 | |
| {
 | |
| 	return (TBaseServerApp&)*wxTheApp;
 | |
| }
 | |
| 
 | |
| enum 
 | |
| {
 | |
|   SERVER_ID = 1001,
 | |
|   SOCKET_ID = 1002
 | |
| };
 | |
| 
 | |
| 
 | |
| BEGIN_EVENT_TABLE(TBaseServerApp, wxApp)
 | |
|   EVT_SOCKET(SERVER_ID, TBaseServerApp::OnServerEvent)
 | |
|   EVT_SOCKET(SOCKET_ID, TBaseServerApp::OnSocketEvent)
 | |
| END_EVENT_TABLE()
 | |
| 
 | |
| void TBaseServerApp::WriteLog(const wxChar* str) const
 | |
| {
 | |
| 	if (m_log != NULL)
 | |
| 	  *m_log << str << endl;
 | |
| }
 | |
| 
 | |
| wxString TBaseServerApp::GetTempFilename()
 | |
| {
 | |
| 	wxString strTmp;
 | |
| 	if (m_strTempDir.IsEmpty())
 | |
| 	{
 | |
|     const wxString strPrefix = "Srv";
 | |
|     ::wxGetTempFileName(strPrefix, strTmp);
 | |
| 		::wxRemoveFile(strTmp);
 | |
| 		(wxString&)m_strTempDir << ::wxPathOnly(strTmp) << "/";
 | |
| 	}
 | |
| 	strTmp = m_strTempDir;
 | |
|   strTmp += wxString::Format("%s_%d.htm", GetAppName(), m_nTmpCounter);
 | |
|   m_nTmpCounter++;
 | |
| 	if (m_nTmpCounter >= 8)
 | |
| 		m_nTmpCounter = 0;
 | |
| 
 | |
| 	return strTmp;
 | |
| }
 | |
| 
 | |
| TXmlItem& TBaseServerApp::AddLogo(TXmlItem& tr) const 
 | |
| {
 | |
| 	TXmlItem& td = tr.AddChild("td");
 | |
| 	td.SetAttr("width", "15%");
 | |
| 	td.SetAttr("align", "center");
 | |
| 	
 | |
| 	TXmlItem& a = td.AddChild("a");
 | |
| 	a.SetAttr("href", "/index.htm");
 | |
|   
 | |
| 	const wxString gif = GetConfigString("Icon", "euro.gif");
 | |
| 	TXmlItem& img = a.AddChild("img");
 | |
| 	img.SetAttr("src", gif);
 | |
| 	img.SetAttr("border", "0");
 | |
| 	img.SetAttr("alt", "Return to Home Page");
 | |
| 	
 | |
| 	return td;
 | |
| }
 | |
| 
 | |
| TXmlItem& TBaseServerApp::CreatePageBody(TXmlItem& html, wxString header) const
 | |
| {
 | |
|   if (header.IsEmpty())
 | |
| 		header << GetAppName() << " Server";
 | |
| 
 | |
| 	html.SetTag("html");
 | |
| 	
 | |
| 	TXmlItem& head = html.AddChild("head");
 | |
|   TXmlItem& meta1 = head.AddChild("meta");
 | |
| 	meta1.SetAttr("name", "GENERATOR"); 
 | |
| 	meta1.SetAttr("content", GetAppName());
 | |
|   TXmlItem& meta2 = head.AddChild("meta");
 | |
| 	meta2.SetAttr("HTTP-EQUIV", "Content-Type"); 
 | |
| 	meta2.SetAttr("content", "text/html; charset=iso-8859-1");
 | |
|   head.AddChild("title") << header;
 | |
| 
 | |
| 	TXmlItem& body = html.AddChild("body");
 | |
| 	body.SetAttr("bgcolor", "#E0E080");
 | |
| 	body.SetAttr("background", "back.gif");
 | |
| 
 | |
| 	TXmlItem& title = body.AddChild("table");
 | |
| 	title.SetAttr("border", "5"); title.SetAttr("width", "100%");
 | |
| 	TXmlItem& title_tr = title.AddChild("tr");
 | |
| 
 | |
| 	if (m_bRunning)
 | |
|     AddLogo(title_tr);
 | |
|   
 | |
| 	TXmlItem& td1 = title_tr.AddChild("td");
 | |
| 	td1.AddChild("h1").AddChild("center") << header;
 | |
| 	
 | |
| 	if (m_bRunning)
 | |
|     AddLogo(title_tr);
 | |
|   
 | |
| 	body.AddChild("br");
 | |
| 	return body;
 | |
| }
 | |
| 
 | |
| TXmlItem& TBaseServerApp::AddLinkButton(TXmlItem& body, const wxChar* strCaption, const wxChar* strHref) const
 | |
| {
 | |
|   TXmlItem& table = body.AddChild("table");
 | |
| 	table.SetAttr("border", "2"); table.SetAttr("bgcolor", "#C0C0C0");
 | |
| 
 | |
| 	TXmlItem& td = table.AddChild("tr").AddChild("td");
 | |
| 	td.SetAttr("align", "center");
 | |
| 	td.AddChild("a").SetAttr("href", strHref) << strCaption;
 | |
| 	
 | |
| 	return table;
 | |
| }
 | |
| 
 | |
| void TBaseServerApp::MessageBox(const wxChar* caption, const wxChar* msg, wxSocketBase& outs)
 | |
| {
 | |
|   TXmlItem html; 
 | |
| 	TXmlItem& body = CreatePageBody(html); 
 | |
| 
 | |
| 	TXmlItem& table = body.AddChild("center").AddChild("table");
 | |
| 	table.SetAttr("width", "70%"); table.SetAttr("border", "2");
 | |
| 	table.AddChild("caption").AddChild("h2") << caption;
 | |
| 
 | |
| 	TXmlItem& td0 = table.AddChild("tr").AddChild("td");
 | |
| 	td0.SetAttr("align", "justify");
 | |
| 	td0.AddChild("h3") << msg;
 | |
| 
 | |
| 	TXmlItem& td1 = table.AddChild("tr").AddChild("td");
 | |
| 	td1.SetAttr("align", "center");
 | |
| 	AddLinkButton(td1, "Return to main page", "index.htm");
 | |
| 
 | |
| 	const wxString strTmp = GetTempFilename();
 | |
| 	html.Save(strTmp);
 | |
| 	SendFile(strTmp, outs);
 | |
| }
 | |
| 
 | |
| bool TBaseServerApp::CanProcessCommand(wxString& cmd, wxSocketBase& outs)
 | |
| {
 | |
| 	if (cmd.Find("stop.cgi") > 0)
 | |
| 	{
 | |
| 	  m_bRunning = false; // Disable gif on title
 | |
|     MessageBox("Warning!", "Server Stopped", outs);
 | |
|     ExitMainLoop();
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void TBaseServerApp::SendFile(wxString strFilename, wxSocketBase& sock)
 | |
| {
 | |
|   const bool bFound = wxFileExists(strFilename);
 | |
| 	if (!bFound)
 | |
| 	{
 | |
| 		const wxString strTmp = GetTempFilename();
 | |
| 		if (!strTmp.IsEmpty())
 | |
| 		{
 | |
| 			TXmlItem html; 
 | |
| 			TXmlItem& body = CreatePageBody(html); 
 | |
| 			body.AddChild("h1") << "Sorry: File not found!";
 | |
| 			body.AddChild("br");
 | |
| 			body.AddChild("h2") << strFilename;
 | |
| 			html.Save(strTmp);
 | |
| 		}
 | |
|  		strFilename = strTmp;
 | |
| 	}
 | |
| 
 | |
| 	wxFileInputStream inf(strFilename);
 | |
| 
 | |
| 	const size_t nSize = inf.GetSize();
 | |
| 
 | |
| 	WriteLog(wxString::Format("Sending %s (%u bytes)", strFilename, nSize));
 | |
|   
 | |
| 	wxString strType;
 | |
| 	wxSplitPath(strFilename, NULL, NULL, &strType);
 | |
| 	wxFileType* type = wxTheMimeTypesManager->GetFileTypeFromExtension(strType);
 | |
| 	if (type != NULL)
 | |
| 		type->GetMimeType(&strType);
 | |
| 	else
 | |
| 		strType = "text/plain";
 | |
| 	const wxDateTime date = ::wxFileModificationTime(strFilename);
 | |
| 
 | |
| 	wxSocketOutputStream outs(sock);
 | |
| 	if (bFound)
 | |
| 	{
 | |
| 	  outs << "HTTP/1.1 200 OK" << endl;
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
|     outs << "HTTP/1.1 401 Not found" << endl;
 | |
| 	}
 | |
| 	outs << "Server: " << GetAppName() << endl;
 | |
| 	outs << "Host: " << wxGetHostName() << endl;
 | |
| 	if (bFound && m_bRunning)
 | |
| 	{
 | |
| 	  outs << "Connection: keep-alive" << endl;
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		outs << "Connection: close" << endl;
 | |
| 	}
 | |
| 	outs << "Content-Type: " << strType << endl;
 | |
|  	outs << "Content-Length: " << wxString::Format("%u", nSize) << endl;
 | |
| 	outs << "Date: " << date.Format("%#c") << endl;
 | |
|   outs << endl;
 | |
| 
 | |
| 	const size_t BUF_TEMP_SIZE = 1024*64;
 | |
|   char* buf = new char[BUF_TEMP_SIZE];
 | |
|   size_t bytes = BUF_TEMP_SIZE;
 | |
| 	size_t nTotal = 0;
 | |
|   while (bytes == BUF_TEMP_SIZE)
 | |
|   {
 | |
|     bytes = inf.Read(buf, bytes).LastRead();
 | |
|     bytes = outs.Write(buf, bytes).LastWrite();
 | |
| 		// if (bytes == BUF_TEMP_SIZE) wxSleep(1);
 | |
| 		sock.WaitForWrite(5);
 | |
| 		nTotal += bytes;
 | |
|   }
 | |
| 	delete buf;
 | |
| 
 | |
| 	if (nTotal < nSize)
 | |
| 	{
 | |
| 		WriteLog(wxString::Format("I sent %u bytes only.", nTotal));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| const wxChar* TBaseServerApp::GetAppName() const
 | |
| {
 | |
| 	// Pure virtual function
 | |
| 	return "Server";
 | |
| }
 | |
| 
 | |
| void TBaseServerApp::ProcessCommand(wxString cmd, wxSocketBase& outs)
 | |
| {
 | |
| 	// Pure virtual function
 | |
| 	WriteLog("Processing...");
 | |
| 	SendFile("index.htm", outs);
 | |
| }
 | |
| 
 | |
| void TBaseServerApp::OnServerEvent(wxSocketEvent& event)
 | |
| {
 | |
|   wxString s = "OnServerEvent: ";
 | |
| 
 | |
|   switch(event.GetSocketEvent())
 | |
|   {
 | |
|     case wxSOCKET_CONNECTION : s.Append("wxSOCKET_CONNECTION"); break;
 | |
|     default                  : s.Append("Unexpected event!"); break;
 | |
|   }
 | |
| 
 | |
|   WriteLog(s);
 | |
| 
 | |
|   // Accept new connection if there is one in the pending
 | |
|   // connections queue, else exit. We use Accept(FALSE) for
 | |
|   // non-blocking accept (although if we got here, there
 | |
|   // should ALWAYS be a pending connection).
 | |
| 
 | |
|   wxSocketBase* sock = m_server->Accept(FALSE);
 | |
|   if (sock)
 | |
|   {
 | |
|     WriteLog("New client connection accepted");
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     WriteLog("Error: couldn't accept a new connection");
 | |
|     sock->Destroy();
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   sock->SetEventHandler(*this, SOCKET_ID);
 | |
|   sock->SetNotify(wxSOCKET_INPUT_FLAG | wxSOCKET_LOST_FLAG);
 | |
|   sock->Notify(TRUE);
 | |
| }
 | |
| 
 | |
| void TBaseServerApp::OnSocketEvent(wxSocketEvent& event)
 | |
| {
 | |
|   wxString s = "OnSocketEvent: ";
 | |
|   wxSocketBase *sock = event.GetSocket();
 | |
| 
 | |
|   // First, print a message
 | |
|   switch(event.GetSocketEvent())
 | |
|   {
 | |
|     case wxSOCKET_INPUT : s.Append("wxSOCKET_INPUT\n"); break;
 | |
|     case wxSOCKET_LOST  : s.Append("wxSOCKET_LOST\n"); break;
 | |
|     default             : s.Append("Unexpected event !\n"); break;
 | |
|   }
 | |
| 
 | |
|   WriteLog(s);
 | |
| 
 | |
|   // Now we process the event
 | |
|   switch(event.GetSocketEvent())
 | |
|   {
 | |
|     case wxSOCKET_INPUT:
 | |
|     {
 | |
| 			const int nOldFlags = sock->GetFlags();
 | |
|       // We disable input events, so that the test doesn't trigger
 | |
|       // wxSocketEvent again.
 | |
|       sock->SetNotify(wxSOCKET_LOST_FLAG);
 | |
| 			sock->SetFlags(wxSOCKET_NOWAIT); // Non aspettare troppo
 | |
| 
 | |
| 			// Read the data
 | |
| 			const size_t BUFSIZE = 2048;
 | |
| 			wxString str;
 | |
| 			wxChar* buf = str.GetWriteBuf(BUFSIZE);
 | |
| 			memset(buf, 0, BUFSIZE);
 | |
|  			const size_t len = sock->Read(buf, BUFSIZE).LastCount();
 | |
|       str.UngetWriteBuf(len+1);
 | |
| 			
 | |
|       WriteLog(str);
 | |
| 
 | |
|       sock->SetFlags(nOldFlags); // Ripristina flags
 | |
| 			if (CanProcessCommand(str, *sock))
 | |
| 			  ProcessCommand(str, *sock);
 | |
| 
 | |
|       // Enable input events again.
 | |
|       sock->SetNotify(wxSOCKET_LOST_FLAG | wxSOCKET_INPUT_FLAG);
 | |
|       break;
 | |
|     }
 | |
|     case wxSOCKET_LOST:
 | |
|     {
 | |
|       // Destroy() should be used instead of delete wherever possible,
 | |
|       // due to the fact that wxSocket uses 'delayed events' (see the
 | |
|       // documentation for wxPostEvent) and we don't want an event to
 | |
|       // arrive to the event handler (the frame, here) after the socket
 | |
|       // has been deleted. Also, we might be doing some other thing with
 | |
|       // the socket at the same time; for example, we might be in the
 | |
|       // middle of a test or something. Destroy() takes care of all
 | |
|       // this for us.
 | |
| 
 | |
|       WriteLog("Deleting socket.");
 | |
|       sock->Destroy();
 | |
|       break;
 | |
|     }
 | |
|     default: ;
 | |
|   }
 | |
| }
 | |
| 
 | |
| const wxChar* TBaseServerApp::GetConfigName() const
 | |
| {
 | |
| 	return "./servers.ini";
 | |
| }
 | |
| 
 | |
| void TBaseServerApp::SetConfigString(const wxChar* key, const wxChar* val, const wxChar* app) const
 | |
| {
 | |
|   wxIniConfig ini("", "", GetConfigName());
 | |
| 	wxString str;
 | |
| 	if (app == NULL || *app == '\0')
 | |
| 		app = GetAppName();
 | |
| 	str << '/' << app;
 | |
| 	ini.SetPath(str);
 | |
| 	ini.Write(key, val);
 | |
| }
 | |
| 
 | |
| void TBaseServerApp::SetConfigInt(const wxChar* key, int val, const wxChar* app) const
 | |
| {
 | |
| 	wxString str = wxString::Format("%d", val);
 | |
|   SetConfigString(key, str, app);
 | |
| }
 | |
| 
 | |
| wxString TBaseServerApp::GetConfigString(const wxChar* key, const wxChar* def, const wxChar* app) const
 | |
| {
 | |
|   wxIniConfig ini("", "", GetConfigName());
 | |
| 	wxString str;
 | |
| 	if (app == NULL || *app == '\0')
 | |
| 		app = GetAppName();
 | |
| 	str << '/' << app;
 | |
| 	ini.SetPath(str);
 | |
| 	ini.Read(key, &str, def);
 | |
| 	return str;
 | |
| }
 | |
| 
 | |
| int TBaseServerApp::GetConfigInt(const wxChar* key, int def, const wxChar* app) const
 | |
| {
 | |
| 	wxString str = GetConfigString(key, "*", app);
 | |
| 	int val = def;
 | |
| 	if (str != "*")
 | |
| 		val = atoi(str);
 | |
| 	return val;
 | |
| }
 | |
| 
 | |
| bool TBaseServerApp::GetConfigBool(const wxChar* key, bool def, const wxChar* app) const
 | |
| {
 | |
| 	wxString str = GetConfigString(key, "*", app);
 | |
| 	bool val = def;
 | |
| 	if (str != "*")
 | |
| 		val = (str[0] == '1') || (str[0] == 'X') || (str[0] == 'Y');
 | |
| 	return val;
 | |
| }
 | |
| 
 | |
| int TBaseServerApp::GetDefaultPort() const
 | |
| {
 | |
| 	return GetConfigInt("Port", 3883);
 | |
| }
 | |
| 
 | |
| wxString TBaseServerApp::GetLogFileName() const
 | |
| {
 | |
|   return GetConfigString("LogFile");
 | |
| }
 | |
| 
 | |
| wxString TBaseServerApp::GetDocumentRoot() const
 | |
| {
 | |
|   return GetConfigString("DocumentRoot", ".");
 | |
| }
 | |
| 
 | |
| bool TBaseServerApp::OnInit()
 | |
| {
 | |
|   // Create the address - defaults to localhost:0 initially
 | |
|   wxIPV4address addr;
 | |
|   addr.Service(GetDefaultPort());
 | |
|   // Create the socket
 | |
|   m_server = new wxSocketServer(addr);
 | |
| 
 | |
|   // Create the Log file
 | |
| 	wxString str;
 | |
| 	
 | |
| 	str = GetLogFileName();
 | |
|   if (!str.IsEmpty())
 | |
|   	m_log = new wxFileOutputStream(str);
 | |
| 	else
 | |
| 		m_log = NULL;
 | |
| 
 | |
| 	m_nTmpCounter = 0;
 | |
| 	
 | |
| 	// We use Ok() here to see if the server is really listening
 | |
|   str.Empty();
 | |
|   if (m_server->Ok())
 | |
|   {
 | |
| 		// Setup the event handler and subscribe to connection events
 | |
| 		m_server->SetEventHandler(*this, SERVER_ID);
 | |
| 		m_server->SetNotify(wxSOCKET_CONNECTION_FLAG);
 | |
| 		m_server->Notify(TRUE);
 | |
| 
 | |
| 		str << GetAppName() << " listening on port " << addr.Service();
 | |
| 		SetConfigString("Running", "1");
 | |
| 		m_bRunning = true;
 | |
| 	}
 | |
|   else
 | |
|   {
 | |
| 		str << GetAppName() << " could not listen to port " << addr.Service();
 | |
|   }
 | |
| 	WriteLog(str);
 | |
| 
 | |
| 	bool ok = m_server->Ok();
 | |
| 	if (ok)
 | |
|     ok = Initialization();
 | |
| 
 | |
| 	return ok;
 | |
| }
 | |
| 
 | |
| int TBaseServerApp::OnExit()
 | |
| {
 | |
|   Deinitialization();
 | |
| 
 | |
| 	if (m_server != NULL)
 | |
|   	delete m_server;
 | |
| 	if (m_log != NULL)
 | |
| 	{
 | |
| 		SetConfigString("Running", "");
 | |
| 		wxString str;
 | |
| 		str << GetAppName() << " shutting down.";
 | |
| 		WriteLog(str);
 | |
| 	  delete m_log;
 | |
| 	}
 | |
| 	return wxApp::OnExit();
 | |
| }
 | |
| 
 | |
| wxString TBaseServerApp::UnformatString(const wxString& strFormString) const
 | |
| {
 | |
| 	wxString strResult;
 | |
| 	for (int i = 0; strFormString[i]; i++)
 | |
| 	{
 | |
| 		switch(strFormString[i])
 | |
| 		{
 | |
| 		case '+': 
 | |
| 			strResult += ' '; break;
 | |
| 		case '%':
 | |
| 			{
 | |
| 				const wxChar c = hex2int(strFormString.Mid(i+1, 2));
 | |
| 				strResult += c;
 | |
| 				i += 2;
 | |
| 			}
 | |
| 			break;
 | |
| 		default: 
 | |
| 			strResult += strFormString[i]; break;
 | |
| 		}
 | |
| 	}
 | |
| 	return strResult;
 | |
| }
 | |
| 
 | |
| int TBaseServerApp::ParseArguments(wxString args, THashTable& hashArgs) const
 | |
| {
 | |
|   int uguale = args.Find('=');
 | |
| 	while (uguale > 0)
 | |
| 	{
 | |
| 		wxString name = args.Left(uguale); 
 | |
| 		name.Trim(false); name.Trim(true);
 | |
| 		
 | |
| 		wxString value;
 | |
| 		const int acapo = args.Find('&');
 | |
| 		if (acapo > uguale)
 | |
| 	    value = args.Mid(uguale+1, acapo-uguale-1); 
 | |
| 		else
 | |
| 			value = args.Mid(uguale+1); 
 | |
| 		value.Trim(false); value.Trim(true);
 | |
| 		hashArgs.Put(name, UnformatString(value));
 | |
| 
 | |
| 		if (acapo > 0)
 | |
| 		{
 | |
|       args = args.Mid(acapo+1);
 | |
| 			uguale = args.Find('=');
 | |
| 		}
 | |
| 		else
 | |
| 			uguale = -1;
 | |
| 	}
 | |
| 	return hashArgs.GetCount();
 | |
| }	
 |