| /******************************************************************************* |
| * Copyright (c) 2005, 2007 IBM Corporation and others. |
| * All rights reserved. This program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License v1.0 |
| * which accompanies this distribution, and is available at |
| * http://www.eclipse.org/legal/epl-v10.html |
| * |
| * Contributors: |
| * IBM Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.jst.j2ee.ejb.componentcore.util; |
| |
| import java.util.List; |
| import java.util.Properties; |
| |
| import org.eclipse.core.resources.IProject; |
| import org.eclipse.core.runtime.CoreException; |
| import org.eclipse.core.runtime.IPath; |
| import org.eclipse.core.runtime.NullProgressMonitor; |
| import org.eclipse.emf.common.util.URI; |
| import org.eclipse.emf.ecore.EObject; |
| import org.eclipse.emf.ecore.EPackage; |
| import org.eclipse.emf.ecore.resource.Resource; |
| import org.eclipse.jem.util.emf.workbench.WorkbenchResourceHelperBase; |
| import org.eclipse.jst.j2ee.application.ApplicationPackage; |
| import org.eclipse.jst.j2ee.application.Module; |
| import org.eclipse.jst.j2ee.commonarchivecore.internal.Archive; |
| import org.eclipse.jst.j2ee.commonarchivecore.internal.CommonarchiveFactory; |
| import org.eclipse.jst.j2ee.commonarchivecore.internal.exception.OpenFailureException; |
| import org.eclipse.jst.j2ee.componentcore.EnterpriseArtifactEdit; |
| import org.eclipse.jst.j2ee.ejb.EJBJar; |
| import org.eclipse.jst.j2ee.ejb.EJBResource; |
| import org.eclipse.jst.j2ee.ejb.EjbFactory; |
| import org.eclipse.jst.j2ee.internal.J2EEConstants; |
| import org.eclipse.jst.j2ee.internal.common.CreationConstants; |
| import org.eclipse.jst.j2ee.internal.common.XMLResource; |
| import org.eclipse.jst.j2ee.internal.componentcore.JavaEEBinaryComponentHelper; |
| import org.eclipse.jst.j2ee.internal.ejb.archiveoperations.EJBComponentLoadStrategyImpl; |
| import org.eclipse.jst.j2ee.internal.project.J2EEProjectUtilities; |
| import org.eclipse.jst.j2ee.model.IModelProvider; |
| import org.eclipse.wst.common.componentcore.ArtifactEdit; |
| import org.eclipse.wst.common.componentcore.ComponentCore; |
| import org.eclipse.wst.common.componentcore.ModuleCoreNature; |
| import org.eclipse.wst.common.componentcore.UnresolveableURIException; |
| import org.eclipse.wst.common.componentcore.internal.ArtifactEditModel; |
| import org.eclipse.wst.common.componentcore.internal.BinaryComponentHelper; |
| import org.eclipse.wst.common.componentcore.internal.StructureEdit; |
| import org.eclipse.wst.common.componentcore.internal.impl.ModuleURIUtil; |
| import org.eclipse.wst.common.componentcore.internal.util.IArtifactEditFactory; |
| import org.eclipse.wst.common.componentcore.resources.IVirtualComponent; |
| import org.eclipse.wst.common.componentcore.resources.IVirtualFolder; |
| import org.eclipse.wst.common.componentcore.resources.IVirtualReference; |
| import org.eclipse.wst.common.componentcore.resources.IVirtualResource; |
| import org.eclipse.wst.common.internal.emfworkbench.WorkbenchResourceHelper; |
| |
| /** |
| * <p> |
| * EJBArtifactEdit obtains a EJB Deployment Descriptor metamodel specifec data from a |
| * {@see org.eclipse.jst.j2ee.ejb.EJBResource} which stores the metamodel. The |
| * {@see org.eclipse.jst.j2ee.ejb.EJBResource} is retrieved from the |
| * {@see org.eclipse.wst.common.modulecore.ArtifactEditModel} using a constant {@see |
| * J2EEConstants#EJBJAR_DD_URI_OBJ}. The defined methods extract data or manipulate the contents of |
| * the underlying resource. |
| * </p> |
| * |
| */ |
| public class EJBArtifactEdit extends EnterpriseArtifactEdit implements IArtifactEditFactory { |
| |
| private static final String EJB_CONTENT_TYPE = "org.eclipse.jst.j2ee.ejbDD"; |
| /** |
| * <p> |
| * Identifier used to link EJBArtifactEdit to a EJBEditAdapterFactory {@see |
| * EJBEditAdapterFactory} stored in an AdapterManger (@see AdapterManager) |
| * </p> |
| */ |
| |
| public static final Class ADAPTER_TYPE = EJBArtifactEdit.class; |
| |
| /** |
| * |
| */ |
| public EJBArtifactEdit() { |
| super(); |
| } |
| |
| public EJBArtifactEdit(IVirtualComponent aModule) { |
| super(aModule); |
| } |
| |
| protected BinaryComponentHelper initBinaryComponentHelper(IVirtualComponent binaryModule) { |
| return new JavaEEBinaryComponentHelper(binaryModule); |
| } |
| |
| /** |
| * @param aHandle |
| * @param toAccessAsReadOnly |
| * @throws IllegalArgumentException |
| */ |
| public EJBArtifactEdit(IProject aProject, boolean toAccessAsReadOnly) throws IllegalArgumentException { |
| super(aProject, toAccessAsReadOnly); |
| } |
| |
| public EJBArtifactEdit(IProject aProject, boolean toAccessAsReadOnly, boolean forCreate) throws IllegalArgumentException { |
| super(aProject, toAccessAsReadOnly, forCreate, J2EEProjectUtilities.EJB); |
| } |
| |
| /** |
| * <p> |
| * Creates an instance facade for the given {@see ArtifactEditModel}. |
| * </p> |
| * |
| * @param anArtifactEditModel |
| */ |
| public EJBArtifactEdit(ArtifactEditModel model) { |
| super(model); |
| } |
| |
| /** |
| * <p> |
| * Creates an instance facade for the given {@see ArtifactEditModel} |
| * </p> |
| * |
| * <p> |
| * Note: This method is for internal use only. Clients should not call this method. |
| * </p> |
| * |
| * @param aNature |
| * A non-null {@see ModuleCoreNature}for an accessible project |
| * @param aModule |
| * A non-null {@see WorkbenchComponent}pointing to a module from the given |
| * {@see ModuleCoreNature} |
| */ |
| public EJBArtifactEdit(ModuleCoreNature aNature, IVirtualComponent aModule, boolean toAccessAsReadOnly) { |
| super(aNature, aModule, toAccessAsReadOnly); |
| } |
| |
| /** |
| * |
| * @return EJBResource from (@link getDeploymentDescriptorResource()) |
| * |
| */ |
| |
| public EJBResource getEJBJarXmiResource() { |
| verifyOperationSupported(); |
| return (EJBResource) getDeploymentDescriptorResource(); |
| } |
| |
| /** |
| * Experimental API subject to change |
| * |
| * @return IVirtualFolder that contains the deployment descriptor resource. |
| * @throws CoreException |
| */ |
| |
| public IVirtualFolder getDeploymentDescriptorFolder() throws CoreException { |
| IVirtualResource[] resources = getComponent().getRootFolder().members(); |
| if (resources != null && resources.length > 0) { |
| for (int i = 0; i < resources.length; i++) { |
| IVirtualResource resource = resources[i]; |
| if (resource.getType() == IVirtualResource.FOLDER) { |
| IVirtualFolder folder = (IVirtualFolder) resource; |
| IVirtualResource ddResource = folder.findMember(J2EEConstants.EJBJAR_DD_SHORT_NAME); |
| if (ddResource != null) |
| return folder; |
| } |
| } |
| } |
| return null; |
| } |
| |
| /** |
| * Experimental API subject to change |
| * |
| * @return String form of the ejbModule folder path relative to the full path. |
| * @throws CoreException |
| */ |
| |
| public String getEjbModuleRelative(String fullPath) throws CoreException { |
| if (fullPath != null) { |
| if (getDeploymentDescriptorFolder() != null) { |
| String modulePath; |
| if (fullPath.indexOf('/') == 0) |
| modulePath = getDeploymentDescriptorFolder().getProjectRelativePath().removeTrailingSeparator().removeLastSegments(1).toOSString(); |
| else |
| modulePath = getDeploymentDescriptorFolder().getProjectRelativePath().removeTrailingSeparator().removeLastSegments(1).makeRelative().toOSString(); |
| int indx = (fullPath.indexOf(modulePath) + modulePath.length() + 1); |
| if (fullPath.indexOf(modulePath) != -1) |
| return fullPath.substring(indx); |
| } |
| } |
| return fullPath; |
| } |
| |
| |
| /** |
| * <p> |
| * Retrieves J2EE version information from EJBResource. |
| * </p> |
| * |
| * @return an integer representation of a J2EE Spec version |
| * |
| */ |
| |
| public int getJ2EEVersion() { |
| verifyOperationSupported(); |
| return getEJBJarXmiResource().getJ2EEVersionID(); |
| } |
| |
| /** |
| * <p> |
| * Checks is a EJB Client Jar exists for the ejb module project |
| * </p> |
| * |
| * @return boolean |
| * |
| */ |
| |
| public boolean hasEJBClientJARProject() { |
| verifyOperationSupported(); |
| |
| if (getEJBClientJarModule() != null) |
| return true; |
| return false; |
| } |
| |
| /** |
| * <p> |
| * Creates a new EJB module |
| * </p> |
| * |
| * @return |
| */ |
| public Module createNewModule() { |
| verifyOperationSupported(); |
| if (isBinary()) { |
| throwAttemptedBinaryEditModelAccess(); |
| } |
| return ((ApplicationPackage) EPackage.Registry.INSTANCE.getEPackage(ApplicationPackage.eNS_URI)).getApplicationFactory().createEjbModule(); |
| } |
| |
| |
| /** |
| * @param project |
| * @return WorkbenchComponent |
| */ |
| public IVirtualComponent getEJBClientJarModule() { |
| verifyOperationSupported(); |
| EJBJar jar = getEJBJar(); |
| IVirtualComponent ejbComponent, ejbClientComponent = null; |
| ejbComponent = ComponentCore.createComponent(getProject()); |
| if (ejbComponent == null) |
| return null; |
| Properties props = ejbComponent.getMetaProperties(); |
| String clientCompName = props.getProperty(CreationConstants.EJB_CLIENT_NAME); |
| if (clientCompName != null && !clientCompName.equals("")) { //$NON-NLS-1$ |
| IVirtualReference vRef = ejbComponent.getReference(clientCompName); |
| if (vRef != null) |
| ejbClientComponent = vRef.getReferencedComponent(); |
| } else { |
| String clientJAR = null; |
| if (jar != null) |
| clientJAR = jar.getEjbClientJar(); |
| if (clientJAR != null) { |
| clientJAR = clientJAR.substring(0, clientJAR.length() - 4); |
| ejbComponent = ComponentCore.createComponent(getProject()); |
| if (ejbComponent == null) |
| return null; |
| IVirtualReference ref = ejbComponent.getReference(clientJAR); |
| if (ref != null) |
| ejbClientComponent = ref.getReferencedComponent(); |
| } |
| } |
| return ejbClientComponent; |
| } |
| |
| |
| /** |
| * <p> |
| * Retrieves the underlying resource from the ArtifactEditModel using defined URI. |
| * </p> |
| * |
| * @return Resource |
| * |
| */ |
| |
| public Resource getDeploymentDescriptorResource() { |
| verifyOperationSupported(); |
| if (isBinary()) { |
| return getBinaryComponentHelper().getResource(J2EEConstants.EJBJAR_DD_URI_OBJ); |
| } |
| return getArtifactEditModel().getResource(J2EEConstants.EJBJAR_DD_URI_OBJ); |
| } |
| |
| /** |
| * |
| * @return EJBJar from (@link getDeploymentDescriptorRoot()) |
| * |
| */ |
| public EJBJar getEJBJar() { |
| verifyOperationSupported(); |
| return (EJBJar) getDeploymentDescriptorRoot(); |
| } |
| |
| /** |
| * <p> |
| * Obtains the EJBJar (@see EJBJar) root object from the EJBResource. If the root object does |
| * not exist, then one is created (@link addEJBJarIfNecessary(getEJBJarXmiResource())). The root |
| * object contains all other resource defined objects. |
| * </p> |
| * |
| * @return EObject |
| * |
| */ |
| public EObject getDeploymentDescriptorRoot() { |
| verifyOperationSupported(); |
| Resource res = getDeploymentDescriptorResource(); |
| List contents = res.getContents(); |
| if (contents.size() > 0) |
| return (EObject) contents.get(0); |
| if (isBinary()) { |
| return null; |
| } |
| if (!WorkbenchResourceHelper.getFile(res).exists()) |
| return null; |
| addEJBJarIfNecessary((EJBResource) getDeploymentDescriptorResource()); |
| return (EObject) contents.get(0); |
| } |
| |
| /** |
| * Returns the deployment descriptor type of the EJB module. |
| * |
| * @return int |
| */ |
| |
| public int getDeploymentDescriptorType() { |
| return XMLResource.EJB_TYPE; |
| } |
| |
| /** |
| * <p> |
| * Creates a deployment descriptor root object (EJBJar) and populates with data. Adds the root |
| * object to the deployment descriptor resource. |
| * </p> |
| * |
| * <p> |
| * |
| * @param aModule |
| * A non-null pointing to a {@see XMLResource} Note: This method is typically used |
| * for JUNIT - move? |
| * </p> |
| */ |
| protected void addEJBJarIfNecessary(XMLResource aResource) { |
| if (isBinary()) { |
| throwAttemptedBinaryEditModelAccess(); |
| } |
| if (aResource != null) { |
| if (aResource.getContents() == null || aResource.getContents().isEmpty()) { |
| EJBJar ejbJar = EjbFactory.eINSTANCE.createEJBJar(); |
| aResource.getContents().add(ejbJar); |
| } |
| EJBJar ejbJar = (EJBJar) aResource.getContents().get(0); |
| URI moduleURI = getArtifactEditModel().getModuleURI(); |
| try { |
| ejbJar.setDisplayName(StructureEdit.getDeployedName(moduleURI)); |
| } catch (UnresolveableURIException e) { |
| // Ignore |
| } |
| aResource.setID(ejbJar, J2EEConstants.EJBJAR_ID); |
| // TODO add more mandatory elements |
| try { |
| if (aResource.isModified()) |
| aResource.saveIfNecessary(); |
| } catch (Exception e) { |
| // TODO Auto-generated catch block |
| e.printStackTrace(); |
| } |
| } |
| } |
| |
| /** |
| * <p> |
| * Returns an instance facade to manage the underlying edit model for the given |
| * {@see WorkbenchComponent}. Instances of ArtifactEdit that are returned through this method |
| * must be {@see #dispose()}ed of when no longer in use. |
| * </p> |
| * <p> |
| * Use to acquire an ArtifactEdit facade for a specific {@see WorkbenchComponent} that |
| * will not be used for editing. Invocations of any save*() API on an instance returned from |
| * this method will throw exceptions. |
| * </p> |
| * <p> |
| * <b>The following method may return null. </b> |
| * </p> |
| * |
| * @param aModule |
| * A valid {@see WorkbenchComponent} with a handle that resolves to an |
| * accessible project in the workspace |
| * @return An instance of ArtifactEdit that may only be used to read the underlying content |
| * model |
| */ |
| public static EJBArtifactEdit getEJBArtifactEditForRead(IProject aProject) { |
| EJBArtifactEdit artifactEdit = null; |
| try { |
| if (isValidEJBModule(ComponentCore.createComponent(aProject))) |
| artifactEdit = new EJBArtifactEdit(aProject, true, false); |
| } catch (Exception e) { |
| artifactEdit = null; |
| } |
| return artifactEdit; |
| } |
| |
| /** |
| * <p> |
| * Returns an instance facade to manage the underlying edit model for the given |
| * {@see WorkbenchComponent}. Instances of ArtifactEdit that are returned through this method |
| * must be {@see #dispose()}ed of when no longer in use. |
| * </p> |
| * <p> |
| * Use to acquire an ArtifactEdit facade for a specific {@see WorkbenchComponent} that |
| * will be used for editing. |
| * </p> |
| * <p> |
| * <b>The following method may return null. </b> |
| * </p> |
| * |
| * @param aModule |
| * A valid {@see WorkbenchComponent} with a handle that resolves to an |
| * accessible project in the workspace |
| * @return An instance of ArtifactEdit that may be used to modify and persist changes to the |
| * underlying content model |
| */ |
| public static EJBArtifactEdit getEJBArtifactEditForWrite(IProject aProject) { |
| EJBArtifactEdit artifactEdit = null; |
| try { |
| if (isValidEJBModule(ComponentCore.createComponent(aProject))) |
| artifactEdit = new EJBArtifactEdit(aProject, false, false); |
| } catch (Exception e) { |
| artifactEdit = null; |
| } |
| return artifactEdit; |
| } |
| |
| |
| /** |
| * <p> |
| * Returns an instance facade to manage the underlying edit model for the given |
| * {@see WorkbenchComponent}. Instances of EJBArtifactEdit that are returned through this |
| * method must be {@see #dispose()}ed of when no longer in use. |
| * </p> |
| * <p> |
| * Use to acquire an EJBArtifactEdit facade for a specific {@see WorkbenchComponent} that |
| * will not be used for editing. Invocations of any save*() API on an instance returned from |
| * this method will throw exceptions. |
| * </p> |
| * <p> |
| * <b>This method may return null. </b> |
| * </p> |
| * |
| * <p> |
| * Note: This method is for internal use only. Clients should not call this method. |
| * </p> |
| * |
| * @param aModule |
| * A valid {@see WorkbenchComponent} with a handle that resolves to an |
| * accessible project in the workspace |
| * @return An instance of EJBArtifactEdit that may only be used to read the underlying content |
| * model |
| * @throws UnresolveableURIException |
| * could not resolve uri. |
| */ |
| public static EJBArtifactEdit getEJBArtifactEditForRead(IVirtualComponent aModule) { |
| if (aModule == null) |
| return null; |
| if (aModule.isBinary()) { |
| return new EJBArtifactEdit(aModule); |
| } |
| return getEJBArtifactEditForRead(aModule.getProject()); |
| } |
| |
| /** |
| * <p> |
| * Returns an instance facade to manage the underlying edit model for the given |
| * {@see WorkbenchComponent}. Instances of EJBArtifactEdit that are returned through this |
| * method must be {@see #dispose()}ed of when no longer in use. |
| * </p> |
| * <p> |
| * Use to acquire an EJBArtifactEdit facade for a specific {@see WorkbenchComponent} that |
| * will be used for editing. |
| * </p> |
| * <p> |
| * <b>This method may return null. </b> |
| * </p> |
| * |
| * <p> |
| * Note: This method is for internal use only. Clients should not call this method. |
| * </p> |
| * |
| * @param aModule |
| * A valid {@see WorkbenchComponent} with a handle that resolves to an |
| * accessible project in the workspace |
| * @return An instance of EJBArtifactEdit that may be used to modify and persist changes to the |
| * underlying content model |
| */ |
| public static EJBArtifactEdit getEJBArtifactEditForWrite(IVirtualComponent aModule) { |
| if (aModule == null || aModule.isBinary()) |
| return null; |
| return getEJBArtifactEditForWrite(aModule.getProject()); |
| } |
| |
| /** |
| * @param module |
| * A {@see WorkbenchComponent} |
| * @return True if the supplied module |
| * {@see ArtifactEdit#isValidEditableModule(WorkbenchComponent)}and the moduleTypeId is |
| * a JST module |
| */ |
| public static boolean isValidEJBModule(IVirtualComponent aModule) throws UnresolveableURIException { |
| if (!isValidEditableModule(aModule)) |
| return false; |
| /* and match the JST_EJB_MODULE type */ |
| return J2EEProjectUtilities.isEJBProject(aModule.getProject()); |
| } |
| |
| /* |
| * (non-Javadoc) |
| * |
| * @see org.eclipse.jst.j2ee.internal.modulecore.util.EnterpriseArtifactEdit#createModelRoot() |
| */ |
| public EObject createModelRoot() { |
| verifyOperationSupported(); |
| if (isBinary()) { |
| throwAttemptedBinaryEditModelAccess(); |
| } |
| |
| return createModelRoot(getJ2EEVersion()); |
| } |
| |
| /* |
| * (non-Javadoc) |
| * |
| * @see org.eclipse.jst.j2ee.internal.modulecore.util.EnterpriseArtifactEdit#createModelRoot(int) |
| */ |
| public EObject createModelRoot(int version) { |
| verifyOperationSupported(); |
| if (isBinary()) { |
| throwAttemptedBinaryEditModelAccess(); |
| } |
| |
| EJBResource res = (EJBResource) getDeploymentDescriptorResource(); |
| res.setModuleVersionID(version); |
| addEJBJarIfNecessary(res); |
| return ((EJBResource) getDeploymentDescriptorResource()).getRootObject(); |
| } |
| |
| public ArtifactEdit createArtifactEditForRead(IVirtualComponent aComponent) { |
| return getEJBArtifactEditForRead(aComponent); |
| } |
| |
| public ArtifactEdit createArtifactEditForWrite(IVirtualComponent aComponent) { |
| return getEJBArtifactEditForWrite(aComponent); |
| } |
| |
| public Archive asArchive(boolean includeSource, boolean includeClasspathComponents) throws OpenFailureException { |
| if (isBinary()) { |
| JavaEEBinaryComponentHelper helper = (JavaEEBinaryComponentHelper)getBinaryComponentHelper(); |
| return helper.accessLegacyArchive(); |
| } else { |
| EJBComponentLoadStrategyImpl loader = new EJBComponentLoadStrategyImpl(getComponent(), includeClasspathComponents); |
| loader.setExportSource(includeSource); |
| String uri = ModuleURIUtil.getHandleString(getComponent()); |
| return CommonarchiveFactory.eINSTANCE.openEJBJarFile(loader, uri); |
| } |
| } |
| |
| public static void createDeploymentDescriptor(IProject project, int version) { |
| EnterpriseArtifactEdit ejbEdit = new EJBArtifactEdit(project, false, true); |
| try { |
| ejbEdit.createModelRoot(version); |
| ejbEdit.save(null); |
| } finally { // Make sure new resource is removed - the uri used for creation shouldn't be cached |
| Resource newRes = ejbEdit.getDeploymentDescriptorResource(); |
| WorkbenchResourceHelperBase.getResourceSet(project).getResources().remove(newRes); |
| newRes.unload(); |
| ejbEdit.dispose(); |
| } |
| } |
| public IModelProvider create(IProject project) { |
| return (IModelProvider)getEJBArtifactEditForRead(project); |
| } |
| |
| public IModelProvider create(IVirtualComponent component) { |
| return (IModelProvider)getEJBArtifactEditForRead(component); |
| } |
| public void modify(Runnable runnable, IPath modelPath) { |
| setWritableEdit(getEJBArtifactEditForWrite(getProject())); |
| try{ |
| runnable.run(); |
| if( getWritableEdit() != null ){ |
| // Always save regardless of resource path passed - Artifactedits save resources as a unit |
| getWritableEdit().saveIfNecessary( new NullProgressMonitor() ); |
| } |
| |
| } finally { //Properly dispose the write artifact edit |
| getWritableEdit().dispose(); |
| setWritableEdit(null); |
| } |
| } |
| |
| |
| protected String getContentTypeDescriber() { |
| return EJB_CONTENT_TYPE; |
| } |
| protected URI getRootURI() { |
| return J2EEConstants.EJBJAR_DD_URI_OBJ; |
| } |
| |
| } |