/*****************************************************************
|
| Platinum - Frame Server
|
| Copyright (c) 2004-2010, Plutinosoft, LLC.
| All rights reserved.
| http://www.plutinosoft.com
|
| 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.
|
| OEMs, ISVs, VARs and other distributors that combine and
| distribute commercially licensed software with Platinum software
| and do not wish to distribute the source code for the commercially
| licensed software under version 2, or (at your option) any later
| version, of the GNU General Public License (the "GPL") must enter
| into a commercial license agreement with Plutinosoft, LLC.
| licensing@plutinosoft.com
|
| 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; see the file LICENSE.txt. If not, write to
| the Free Software Foundation, Inc.,
| 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
| http://www.gnu.org/licenses/gpl-2.0.html
|
****************************************************************/
/*----------------------------------------------------------------------
| includes
+---------------------------------------------------------------------*/
#include "PltFrameStream.h"
#include "PltFrameServer.h"
#include "PltUtilities.h"
NPT_SET_LOCAL_LOGGER("platinum.media.server.frame")
/*----------------------------------------------------------------------
| constants
+---------------------------------------------------------------------*/
#define BOUNDARY "BOUNDARYGOAWAY"
/*----------------------------------------------------------------------
| PLT_SocketPolicyServer
+---------------------------------------------------------------------*/
class PLT_SocketPolicyServer : public NPT_Thread
{
public:
PLT_SocketPolicyServer(const char* policy,
NPT_IpPort port = 0,
const char* authorized_ports = "5900") :
m_Socket(NPT_SOCKET_FLAG_CANCELLABLE),
m_Policy(policy),
m_Port(port),
m_AuthorizedPorts(authorized_ports),
m_Aborted(false) {}
~PLT_SocketPolicyServer() {
Stop();
}
NPT_Result Start() {
NPT_Result result = NPT_FAILURE;
// bind
// randomly try a port for our http server
int retries = 100;
do {
int random = NPT_System::GetRandomInteger();
NPT_IpPort port = (unsigned short)(50000 + (random % 15000));
result = m_Socket.Bind(
NPT_SocketAddress(NPT_IpAddress::Any, m_Port?m_Port:port),
false);
if (NPT_SUCCEEDED(result) || m_Port)
break;
} while (--retries > 0);
if (NPT_FAILED(result) || retries == 0) return NPT_FAILURE;
// remember that we're bound
NPT_SocketInfo info;
m_Socket.GetInfo(info);
m_Port = info.local_address.GetPort();
return NPT_Thread::Start();
}
NPT_Result Stop() {
m_Aborted = true;
m_Socket.Cancel();
return Wait();
}
void Run() {
do {
// wait for a connection
NPT_Socket* client = NULL;
NPT_LOG_FINE_1("waiting for connection on port %d...", m_Port);
NPT_Result result = m_Socket.WaitForNewClient(client, NPT_TIMEOUT_INFINITE);
if (NPT_FAILED(result) || client == NULL) return;
NPT_SocketInfo client_info;
client->GetInfo(client_info);
NPT_LOG_FINE_2("client connected (%s -> %s)",
client_info.local_address.ToString().GetChars(),
client_info.remote_address.ToString().GetChars());
// get the output stream
NPT_OutputStreamReference output;
client->GetOutputStream(output);
// generate policy based on our current IP
NPT_String policy = "";
policy += "";
policy += "";
policy += "";
NPT_MemoryStream* mem_input = new NPT_MemoryStream();
mem_input->Write(policy.GetChars(), policy.GetLength());
NPT_InputStreamReference input(mem_input);
NPT_StreamToStreamCopy(*input, *output);
delete client;
} while (!m_Aborted);
}
NPT_TcpServerSocket m_Socket;
NPT_String m_Policy;
NPT_IpPort m_Port;
NPT_String m_AuthorizedPorts;
bool m_Aborted;
};
/*----------------------------------------------------------------------
| PLT_HttpStreamRequestHandler::SetupResponse
+---------------------------------------------------------------------*/
NPT_Result
PLT_HttpStreamRequestHandler::SetupResponse(NPT_HttpRequest& request,
const NPT_HttpRequestContext& context,
NPT_HttpResponse& response)
{
PLT_LOG_HTTP_REQUEST(NPT_LOG_LEVEL_FINE, "PLT_HttpStreamRequestHandler::SetupResponse:", &request);
if (request.GetMethod().Compare("GET") &&
request.GetMethod().Compare("HEAD")) {
return NPT_FAILURE;
}
NPT_Reference buffer;
if (!m_StreamValidator.OnNewRequestAccept(request, context, response, buffer)) {
return NPT_ERROR_NO_SUCH_ITEM;
}
response.SetProtocol(NPT_HTTP_PROTOCOL_1_0);
response.GetHeaders().SetHeader(NPT_HTTP_HEADER_CONNECTION, "close");
response.GetHeaders().SetHeader("Cache-Control", "no-store, no-cache, must-revalidate, pre-check=0, post-check=0, max-age=0");
response.GetHeaders().SetHeader("Pragma", "no-cache");
response.GetHeaders().SetHeader("Expires", "Tue, 4 Jan 2000 02:43:05 GMT");
// HEAD request has no entity or if status code is not 2xx
if (!request.GetMethod().Compare("HEAD") || response.GetStatusCode()/100 != 2)
return NPT_SUCCESS;
NPT_HttpEntity* entity = response.GetEntity();
NPT_CHECK_POINTER_FATAL(entity);
entity->SetContentType("multipart/x-mixed-replace;boundary=" BOUNDARY);
NPT_InputStreamReference body(new PLT_InputFrameStream(buffer, BOUNDARY));
entity->SetInputStream(body, false);
return NPT_SUCCESS;
}
/*----------------------------------------------------------------------
| PLT_FrameServer::PLT_FrameServer
+---------------------------------------------------------------------*/
PLT_FrameServer::PLT_FrameServer(const char* resource_name,
PLT_StreamValidator& stream_validator,
NPT_IpAddress address,
NPT_UInt16 port,
bool policy_server_enabled) :
PLT_HttpServer(address, port, false),
m_PolicyServer(NULL),
m_StreamValidator(stream_validator),
m_PolicyServerEnabled(policy_server_enabled)
{
NPT_String resource(resource_name);
resource.Trim("/\\");
AddRequestHandler(
new PLT_HttpStreamRequestHandler(stream_validator),
"/" + resource,
true,
true);
}
/*----------------------------------------------------------------------
| PLT_FrameServer::~PLT_FrameServer
+---------------------------------------------------------------------*/
PLT_FrameServer::~PLT_FrameServer()
{
delete m_PolicyServer;
}
/*----------------------------------------------------------------------
| PLT_FrameServer::Start
+---------------------------------------------------------------------*/
NPT_Result
PLT_FrameServer::Start()
{
// start main server so we can get the listening port
NPT_CHECK_SEVERE(PLT_HttpServer::Start());
// start the xml socket policy server for flash
if (m_PolicyServerEnabled) {
m_PolicyServer = new PLT_SocketPolicyServer(
"",
8989,
"5900,"+NPT_String::FromInteger(GetPort()));
NPT_CHECK_SEVERE(m_PolicyServer->Start());
}
return NPT_SUCCESS;
}