All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.red5.server.net.servlet.AMFGatewayServlet Maven / Gradle / Ivy

/*
 * RED5 Open Source Media Server - https://github.com/Red5/
 * 
 * Copyright 2006-2016 by respective authors (see below). All rights reserved.
 * 
 * Licensed under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.red5.server.net.servlet;

import java.io.IOException;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.mina.core.buffer.IoBuffer;
import org.red5.logging.Red5LoggerFactory;
import org.red5.server.api.IContext;
import org.red5.server.api.IServer;
import org.red5.server.api.Red5;
import org.red5.server.api.remoting.IRemotingConnection;
import org.red5.server.api.scope.IGlobalScope;
import org.red5.server.api.scope.IScope;
import org.red5.server.api.service.IServiceInvoker;
import org.red5.server.net.remoting.RemotingConnection;
import org.red5.server.net.remoting.codec.RemotingCodecFactory;
import org.red5.server.net.remoting.message.RemotingCall;
import org.red5.server.net.remoting.message.RemotingPacket;
import org.slf4j.Logger;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.WebApplicationContextUtils;

/**
 * Servlet that handles remoting requests.
 * 
 * @author The Red5 Project
 * @author Luke Hubbard ([email protected])
 * @author Paul Gregoire ([email protected])
 */
public class AMFGatewayServlet extends HttpServlet {

    private static final long serialVersionUID = 7174018823796785619L;

    /**
     * Logger
     */
    protected Logger log = Red5LoggerFactory.getLogger(AMFGatewayServlet.class);

    /**
     * AMF MIME type
     */
    public static final String APPLICATION_AMF = "application/x-amf";

    /**
     * Web app context
     */
    protected transient WebApplicationContext webAppCtx;

    /**
     * Red5 server instance
     */
    protected transient IServer server;

    /**
     * Remoting codec factory
     */
    protected transient RemotingCodecFactory codecFactory;

    /**
     * Request attribute holding the Red5 connection object
     */
    private static final String CONNECTION = "red5.remotingConnection";

    /** {@inheritDoc} */
    @Override
    public void init() throws ServletException {
    }

    /** {@inheritDoc} */
    @Override
    public void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        log.debug("Servicing Request");
        if (codecFactory == null) {
            ServletContext ctx = getServletContext();
            log.debug("Context path: {}", ctx.getContextPath());
            //attempt to lookup the webapp context		
            webAppCtx = WebApplicationContextUtils.getRequiredWebApplicationContext(ctx);
            //now try to look it up as an attribute
            if (webAppCtx == null) {
                log.debug("Webapp context was null, trying lookup as attr.");
                webAppCtx = (WebApplicationContext) ctx.getAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE);
            }
            //lookup the server and codec factory
            if (webAppCtx != null) {
                server = (IServer) webAppCtx.getBean("red5.server");
                codecFactory = (RemotingCodecFactory) webAppCtx.getBean("remotingCodecFactory");
            } else {
                log.debug("No web context");
            }
        }
        log.debug("Remoting request {} {}", req.getContextPath(), req.getServletPath());
        if (APPLICATION_AMF.equals(req.getContentType())) {
            serviceAMF(req, resp);
        } else {
            resp.getWriter().write("Red5 : Remoting Gateway");
        }
    }

    /**
     * Return the global scope to use for the given request.
     * 
     * @param req
     *            http request
     * @return scope
     */
    protected IGlobalScope getGlobalScope(HttpServletRequest req) {
        String path = req.getContextPath() + req.getServletPath();
        log.debug("getGlobalScope path: {}", path);
        if (path.startsWith("/")) {
            path = path.substring(1);
        } else {
            log.debug("Path length: {} Servlet name length: {}", path.length(), getServletName().length());
            path = path.substring(0, path.length() - getServletName().length() - 1);
        }
        IGlobalScope global = server.lookupGlobal(req.getServerName(), path);
        if (global == null) {
            global = server.lookupGlobal(req.getLocalName(), path);
            if (global == null) {
                global = server.lookupGlobal(req.getLocalAddr(), path);
            }
        }
        return global;
    }

    /**
     * Works out AMF request
     * 
     * @param req
     *            Request
     * @param resp
     *            Response
     * @throws ServletException
     *             Servlet exception
     * @throws IOException
     *             I/O exception
     */
    protected void serviceAMF(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        log.debug("Servicing AMF");
        IRemotingConnection conn = null;
        try {
            RemotingPacket packet = decodeRequest(req);
            if (packet == null) {
                log.error("Packet should not be null");
                return;
            }
            // Provide a valid IConnection in the Red5 object
            final IGlobalScope global = getGlobalScope(req);
            final IContext context = global.getContext();
            final IScope scope = context.resolveScope(global, packet.getScopePath());
            conn = new RemotingConnection(req, scope, packet);
            // Make sure the connection object isn't garbage collected
            req.setAttribute(CONNECTION, conn);
            // set thread local reference
            Red5.setConnectionLocal(conn);
            //fixed so that true is not returned for calls that have failed
            boolean passed = handleRemotingPacket(req, context, scope, packet);
            if (passed) {
                resp.setStatus(HttpServletResponse.SC_OK);
            } else {
                log.warn("At least one invocation failed to execute");
                resp.setStatus(HttpServletResponse.SC_EXPECTATION_FAILED);
            }
            //send our response
            resp.setContentType(APPLICATION_AMF);
            sendResponse(resp, packet);
        } catch (Exception e) {
            log.error("Error handling remoting call", e);
            resp.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
        } finally {
            //ensure the conn attr gets removed
            req.removeAttribute(CONNECTION);
            //unregister the remote connection client
            if (conn != null) {
                ((RemotingConnection) conn).cleanup();
            }
            // clear thread local reference
            Red5.setConnectionLocal(null);
        }
    }

    /**
     * Decode request
     * 
     * @param req
     *            Request
     * @return Remoting packet
     * @throws Exception
     *             General exception
     */
    protected RemotingPacket decodeRequest(HttpServletRequest req) throws Exception {
        log.debug("Decoding request");
        IoBuffer reqBuffer = IoBuffer.allocate(req.getContentLength());
        ServletUtils.copy(req, reqBuffer.asOutputStream());
        reqBuffer.flip();
        RemotingPacket packet = (RemotingPacket) codecFactory.getRemotingDecoder().decode(reqBuffer);
        String path = req.getContextPath();
        if (path == null) {
            path = "";
        }
        if (req.getPathInfo() != null) {
            path += req.getPathInfo();
        }
        // check for header path, this is used by the AMF tunnel servlet
        String headerPath = req.getHeader("Tunnel-request");
        // it is only used if the path is set to root
        if (headerPath != null && path.length() < 1) {
            path = headerPath;
        }
        if (path.length() > 0 && path.charAt(0) == '/') {
            path = path.substring(1);
        }
        log.debug("Path: {} Scope path: {}", path, packet.getScopePath());
        packet.setScopePath(path);
        reqBuffer.free();
        reqBuffer = null;
        return packet;
    }

    /**
     * Handles AMF request by making calls
     * 
     * @param req
     *            Request
     * @param context
     *            context
     * @param scope
     *            scope
     * @param message
     *            Remoting packet
     * @return 
     * true
     * 
* * on success */ protected boolean handleRemotingPacket(HttpServletRequest req, IContext context, IScope scope, RemotingPacket message) { log.debug("Handling remoting packet"); boolean result = true; final IServiceInvoker invoker = context.getServiceInvoker(); for (RemotingCall call : message.getCalls()) { result = invoker.invoke(call, scope); //if we encounter a failure break out if (!result) { break; } } return result; } /** * Sends response to client * * @param resp * Response * @param packet * Remoting packet * @throws Exception * General exception */ protected void sendResponse(HttpServletResponse resp, RemotingPacket packet) throws Exception { log.debug("Sending response"); IoBuffer respBuffer = codecFactory.getRemotingEncoder().encode(packet); if (respBuffer != null) { final ServletOutputStream out = resp.getOutputStream(); resp.setContentLength(respBuffer.limit()); ServletUtils.copy(respBuffer.asInputStream(), out); out.flush(); out.close(); respBuffer.free(); respBuffer = null; } else { log.info("Response buffer was null after encoding"); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy