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

org.glassfish.deployment.common.ClientJarWriter Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2022 Contributors to the Eclipse Foundation
 * Copyright (c) 2012, 2018 Oracle and/or its affiliates. All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */

package org.glassfish.deployment.common;

import com.sun.enterprise.deployment.deploy.shared.InputJarArchive;
import com.sun.enterprise.deployment.deploy.shared.OutputJarArchive;
import com.sun.enterprise.util.io.FileUtils;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.jar.Attributes;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import java.util.jar.Manifest;
import java.util.logging.Level;
import java.util.logging.Logger;

import org.glassfish.api.deployment.DeployCommandParameters;
import org.glassfish.api.deployment.OpsParams;
import org.glassfish.api.deployment.archive.WritableArchiveEntry;
import org.glassfish.deployment.common.Artifacts.FullAndPartURIs;
import org.glassfish.deployment.versioning.VersioningUtils;
import org.glassfish.internal.deployment.ExtendedDeploymentContext;

/**
 * Writes a client JAR file, if one is needed, suitable for downloading.
 * 

* Deployers will use the {@link ClientArtifactsManager} to record files which * they generate that need to be included in the downloadable client JAR. Examples * are EJB stubs, web services artifacts, and JARs related to app clients. * Once all deployers have run through the prepare phase this class can be used to * create the client JAR, collecting those client artifacts into the * generated JAR, and adding the generated client JAR to the list of files * related to this application that are available for download. * * @author tjquinn */ public class ClientJarWriter { private static final Logger LOG = org.glassfish.deployment.common.DeploymentContextImpl.deplLogger; private final ExtendedDeploymentContext deploymentContext; private final String name; private final Map jarFiles = new HashMap<>(); public ClientJarWriter(final ExtendedDeploymentContext deploymentContext) { this.deploymentContext = deploymentContext; name = VersioningUtils.getUntaggedName(deploymentContext.getCommandParameters(DeployCommandParameters.class).name()); } public void run() throws IOException { // Only generate the JAR if we would not already have done so. if (isArtifactsPresent()) { LOG.log(Level.FINE, "Skipping possible client JAR generation because it would already have been done"); return; } final Artifacts downloadableArtifacts = DeploymentUtils.downloadableArtifacts(deploymentContext); final Artifacts generatedArtifacts = DeploymentUtils.generatedArtifacts(deploymentContext); final File clientJarFile = createClientJARIfNeeded(name); if (clientJarFile == null) { LOG.log(Level.FINE, "No client JAR generation is needed."); } else { LOG.log(Level.INFO, "Generated client JAR {0} for possible download of size {1} B", new Object[] {clientJarFile, clientJarFile.length()}); downloadableArtifacts.addArtifact(clientJarFile.toURI(), clientJarFile.getName()); generatedArtifacts.addArtifact(clientJarFile.toURI(), clientJarFile.getName()); } } private boolean isArtifactsPresent() { return deploymentContext.getCommandParameters(OpsParams.class).origin.isArtifactsPresent(); } private File createClientJARIfNeeded(final String appName) throws IOException { LOG.log(Level.FINEST, "createClientJARIfNeeded(appName={0})", appName); final ClientArtifactsManager clientArtifactsManager = ClientArtifactsManager.get(deploymentContext); if (clientArtifactsManager.isEmpty()) { return null; } // Make sure the scratch directories are there. deploymentContext.prepareScratchDirs(); final String generatedClientJARName = generatedClientJARNameAndType(appName); final File generatedClientJARFile = new File(deploymentContext.getScratchDir("xml"), generatedClientJARName); // The app client deployer might have already created the generated JAR file. // In that case we need to merge its contents with what has been // registered with the client artifacts manager. final File movedPreexistingFile; if (generatedClientJARFile.exists()) { try { movedPreexistingFile = mergeContentsToClientArtifactsManager(generatedClientJARFile, clientArtifactsManager); LOG.log(Level.FINEST, "Moved preexisting file: {0}", movedPreexistingFile); } catch (URISyntaxException ex) { throw new IOException(ex); } } else { movedPreexistingFile = null; } // We need our own copy of the artifacts collection because we might // need to add the manifest to it if a manifst is not already in the collection. // The client artifacts manager returns an unalterable collection. final Collection artifacts = new ArrayList<>(clientArtifactsManager.artifacts()); try (OutputJarArchive generatedClientJAR = new OutputJarArchive()) { generatedClientJAR.create(generatedClientJARFile.toURI()); if (isManifestPresent(artifacts)) { LOG.log(Level.FINEST, "The jar file already contains a manifest, no need to generate another."); } else { FullAndPartURIs manifest = generateManifest(); LOG.log(Level.CONFIG, "Generated manifest: {0}", manifest); artifacts.add(manifest); } copyArtifactsToClientJAR(generatedClientJAR, artifacts); return generatedClientJARFile; } catch (IOException e) { if (!generatedClientJARFile.delete()) { generatedClientJARFile.deleteOnExit(); } throw e; } finally { if (movedPreexistingFile != null) { FileUtils.deleteFileNowOrLater(movedPreexistingFile); } } } private File mergeContentsToClientArtifactsManager( final File generatedClientJARFile, final ClientArtifactsManager clientArtifactsManager) throws IOException, URISyntaxException { // First, move the existing JAR to another name so when the caller creates the generated // client JAR it does not overwrite the existing file created by the app client deployer. final File movedGeneratedFile = File.createTempFile(generatedClientJARFile.getName(), ".tmp", generatedClientJARFile.getParentFile()); FileUtils.renameFile(generatedClientJARFile, movedGeneratedFile); try (InputJarArchive existingGeneratedJAR = new InputJarArchive(movedGeneratedFile)) { for (Enumeration e = existingGeneratedJAR.entries(); e.hasMoreElements();) { final String entryName = e.nextElement(); final URI entryURI = new URI("jar", movedGeneratedFile.toURI().toASCIIString() + "!/" + entryName, null); final Artifacts.FullAndPartURIs uris = new Artifacts.FullAndPartURIs(entryURI, entryName); clientArtifactsManager.add(uris); } // Handle the manifest explicitly because the Archive entries() method consciously omits it. final Artifacts.FullAndPartURIs manifestURIs = new Artifacts.FullAndPartURIs( new URI("jar", movedGeneratedFile.toURI().toASCIIString() + "!/" + JarFile.MANIFEST_NAME, null), JarFile.MANIFEST_NAME); clientArtifactsManager.add(manifestURIs); return movedGeneratedFile; } } private boolean isManifestPresent(final Collection artifacts) { for (Artifacts.FullAndPartURIs a : artifacts) { if (a.getPart().toASCIIString().equals(JarFile.MANIFEST_NAME)) { return true; } } return false; } private Artifacts.FullAndPartURIs generateManifest() throws IOException { final File mfFile = File.createTempFile("clientmf", ".MF"); try (OutputStream mfOS = new BufferedOutputStream(new FileOutputStream(mfFile))) { final Manifest mf = new Manifest(); mf.getMainAttributes().put(Attributes.Name.MANIFEST_VERSION, "1.0"); mf.write(mfOS); } return new Artifacts.FullAndPartURIs(mfFile.toURI(), JarFile.MANIFEST_NAME, true); } private static String generatedClientJARNameAndType(final String earName) { return generatedClientJARPrefix(earName) + ".jar"; } private static String generatedClientJARPrefix(final String earName) { return earName + "Client"; } private void copyArtifactsToClientJAR( final OutputJarArchive generatedJar, final Collection artifacts) throws IOException { final Set pathsWrittenToJAR = new HashSet<>(); final StringBuilder copiedFiles = LOG.isLoggable(Level.CONFIG) ? new StringBuilder() : null; for (Artifacts.FullAndPartURIs artifact : artifacts) { // Make sure all ancestor directories are present in the JAR as empty entries. String artPath = artifact.getPart().getRawPath(); int previousSlash = artPath.indexOf('/'); while (previousSlash != -1) { String partialAncestorPath = artPath.substring(0, previousSlash + 1); if (!pathsWrittenToJAR.contains(partialAncestorPath)) { try (WritableArchiveEntry entry = generatedJar.putNextEntry(partialAncestorPath)) { // just an empty entry } pathsWrittenToJAR.add(partialAncestorPath); } previousSlash = artPath.indexOf('/', previousSlash + 1); } try (WritableArchiveEntry entry = generatedJar.putNextEntry(artifact.getPart().toASCIIString()); InputStream is = openInputStream(artifact)) { FileUtils.copy(is, entry); if (copiedFiles != null) { copiedFiles.append(System.lineSeparator()).append(" ").append(artifact.getFull().toASCIIString()) .append(" -> ").append(artifact.getPart().toASCIIString()); } } catch (Exception e) { LOG.log(Level.WARNING, "Could not copy the entry " + artifact.getPart(), e); } finally { if (artifact.isTemporary()) { final File artifactFile = new File(artifact.getFull()); if (!artifactFile.delete()) { artifactFile.deleteOnExit(); } } } } if (copiedFiles != null) { LOG.log(Level.CONFIG, "Copied files: {0}", copiedFiles); } } private InputStream openInputStream(Artifacts.FullAndPartURIs artifact) throws URISyntaxException, IOException { LOG.log(Level.FINE, "openInputStream(artifact={0})", artifact); final URI fullURI = artifact.getFull(); final String scheme = fullURI.getScheme(); if (scheme.equals("file")) { return new FileInputStream(new File(fullURI)); } else if (scheme.equals("jar")) { final String ssp = fullURI.getSchemeSpecificPart(); final URI jarURI = new URI(ssp.substring(0, ssp.indexOf("!/"))); JarFile jf = jarFiles.get(jarURI); if (jf == null) { jf = new JarFile(new File(jarURI)); jarFiles.put(jarURI, jf); } final String entryName = ssp.substring(ssp.indexOf("!/") + 2); final JarEntry jarEntry = jf.getJarEntry(entryName); return jf.getInputStream(jarEntry); } else { throw new IllegalArgumentException(scheme + " != [file,jar]"); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy