Package org.glassfish.appclient.server.core.jws.servedcontent

Source Code of org.glassfish.appclient.server.core.jws.servedcontent.StreamedAutoSignedStaticContent

/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2013-2014 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License").  You
* may not use this file except in compliance with the License.  You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt.  See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license."  If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above.  However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package org.glassfish.appclient.server.core.jws.servedcontent;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.net.URI;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.zip.ZipOutputStream;
import org.glassfish.appclient.server.core.AppClientDeployerHelper;
import org.glassfish.appclient.server.core.jws.AppClientHTTPAdapter;
import static org.glassfish.appclient.server.core.jws.RestrictedContentAdapter.DATE_HEADER_NAME;
import static org.glassfish.appclient.server.core.jws.RestrictedContentAdapter.LAST_MODIFIED_HEADER_NAME;
import org.glassfish.grizzly.http.server.Request;
import org.glassfish.grizzly.http.server.Response;
import org.glassfish.grizzly.http.server.Session;
import org.glassfish.grizzly.http.util.HttpStatus;
import org.glassfish.grizzly.http.util.MimeType;

/**
* Auto-signed content that is delivered not as an on-disk file but as an
* output stream.
* <p>
* This implementation supports the requirement to "sign" a JNLP document by
* including it in a signed JAR so the Java Web Start client can verify that
* the JNLP delivered is the correct document (as compared to the copy in the
* signed JAR delivered to the client).
* <p>
* Most signed JARs served by the Java Web Start support in GlassFish are
* created on-disk by signing their unsigned counterparts once.  Then the on-disk
* signed JAR is served when the client asks for the JAR.  The main JAR has to
* be handled differently to meet the Java Web start security requirements. 
* The signed copy of the main JAR is created on-the-fly by signing the unsigned
* JAR's contents plus the previously-generated JNLP.  The signed output is
* sent directly to the HTTP response's output stream rather than creating a
* temporary file.
* <p>
* We cannot create the signed main JAR once and for all because the JNLP
* document which launches the app might change from one invocation to the next,
* particularly regarding command-line arguments which appear in the JNLP as
* <argument> elements.
*
* @author tjquinn
*/
public class StreamedAutoSignedStaticContent extends AutoSignedContent {

    private static final String SIGNED_JNLP_PATH = "JNLP-INF/APPLICATION.JNLP";
   
    private static final Logger logger = Logger.getLogger(AppClientDeployerHelper.ACC_MAIN_LOGGER, AppClientDeployerHelper.LOG_MESSAGE_RESOURCE);
   
    public StreamedAutoSignedStaticContent(final File unsignedFile,
            final String userProvidedAlias,
            final ASJarSigner jarSigner,
            final String relativeURI,
            final String appName) throws FileNotFoundException {
        super(unsignedFile, null /* signedFile */, userProvidedAlias, jarSigner, relativeURI, appName);
    }
   
    @Override
    public void process(String relativeURIString, Request gReq, Response gResp) throws IOException {
        logger.log(Level.FINE, "Processing main JAR for {0}", gReq.getRequestURI());
        final long now = System.currentTimeMillis();
        gResp.setDateHeader(LAST_MODIFIED_HEADER_NAME, now);
        gResp.setDateHeader(DATE_HEADER_NAME, now);
        gResp.setStatus(HttpStatus.OK_200);
        gResp.setContentType(contentType(unsignedFile()));
       
        /*
         * This is the main JAR, so combine it with the previously-generated
         * and saved main JNLP document.
         */
        final Session session = gReq.getSession();
        final Object jwsObj = session.getAttribute(AppClientHTTPAdapter.GF_JWS_SESSION_CACHED_JNLP_NAME);
        if (jwsObj == null) {
            logger.log(Level.FINE, "Session {0} did not contain cached JNLP", session.getIdInternal());
            throw new NullPointerException();
        }
        if ( ! (jwsObj instanceof byte[])) {
            logger.log(Level.FINE, "Session {0} cached JNLP is not a byte[] as expected", session.getIdInternal());
            throw new IllegalArgumentException(jwsObj.getClass().toString());
        }

        /*
         * We do not know the finished length so we cannot set Content-Length.
         * Instead we'll just write to the output stream and Grizzly will
         * chunk it for us if needed.
         */
        final ZipOutputStream zos = new ZipOutputStream(gResp.getOutputStream());
       
        logger.log(Level.FINE, "Request's session contains cached JNLP");
        final byte[] jnlpContent = (byte[]) jwsObj;
        final Map<String,byte[]> addedContent = new HashMap<String,byte[]>();
        addedContent.put(SIGNED_JNLP_PATH, jnlpContent);
        try {
            jarSigner().signJar(unsignedFile(), zos, userProvidedAlias(), createJWSAttrs(AppClientHTTPAdapter.requestURI(gReq), appName()), addedContent);
            /*
             * Create an on-disk copy of the signed JAR for debugging purposes
             * if logging is detailed enough.
             */
            if (logger.isLoggable(Level.FINEST)) {
                final File debugSignedJARFile = new File(unsignedFile().getAbsolutePath()+".debug");
                final ZipOutputStream dbgZos = new ZipOutputStream(new BufferedOutputStream(new FileOutputStream(debugSignedJARFile)));
                try {
                    jarSigner().signJar(unsignedFile(), dbgZos, userProvidedAlias(), createJWSAttrs(AppClientHTTPAdapter.requestURI(gReq), appName()), addedContent);
                } finally {
                    dbgZos.close();
                }
                logger.log(Level.FINEST, "Created on-disk signed JAR {0}", debugSignedJARFile.getAbsolutePath());
            }
            zos.close();
        } catch (IOException ioex) {
            throw ioex;
        } catch (Exception ex) {
            throw new IOException(ex);
        } finally {
            zos.close();
        }
    }

    @Override
    public boolean isAvailable(URI requestURI) throws IOException {
        return true;
    }
   
   
    private String contentType(final File file) {
        final String path = file.getPath();
        String substr;
        int dot = path.lastIndexOf('.');
        if (dot < 0) {
            substr = file.toString();
            dot = substr.lastIndexOf('.');
        } else {
            substr = path;
        }

        if (dot > 0) {
            String ext = substr.substring(dot + 1);
            String ct = MimeType.get(ext);
            return ct;
        } else {
            return MimeType.get("html");
        }
    }
}
TOP

Related Classes of org.glassfish.appclient.server.core.jws.servedcontent.StreamedAutoSignedStaticContent

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.