/******************************************************************************* * Copyright (c) 2008-2010 Sonatype, Inc. * 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: * Sonatype, Inc. - initial API and implementation *******************************************************************************/ package org.eclipse.m2e.editor.xml.internal; import java.io.File; import java.io.IOException; import java.util.List; import java.util.Stack; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.w3c.dom.Element; import org.w3c.dom.Node; import org.eclipse.core.filebuffers.FileBuffers; import org.eclipse.core.filebuffers.ITextFileBuffer; import org.eclipse.core.filesystem.IFileStore; import org.eclipse.core.resources.IFile; import org.eclipse.core.resources.IProject; import org.eclipse.core.resources.IResource; import org.eclipse.core.resources.ResourcesPlugin; import org.eclipse.core.runtime.CoreException; import org.eclipse.core.runtime.IAdaptable; import org.eclipse.core.runtime.IPath; import org.eclipse.core.runtime.Path; import org.eclipse.jface.text.IDocument; import org.eclipse.jface.text.ITextViewer; import org.eclipse.wst.sse.core.StructuredModelManager; import org.eclipse.wst.sse.core.internal.provisional.IStructuredModel; import org.eclipse.wst.sse.core.internal.provisional.IndexedRegion; import org.eclipse.wst.sse.core.internal.provisional.text.IStructuredDocument; import org.eclipse.wst.xml.core.internal.provisional.document.IDOMModel; import org.apache.maven.model.InputLocation; import org.apache.maven.model.InputSource; import org.apache.maven.project.MavenProject; import org.eclipse.m2e.core.MavenPlugin; import org.eclipse.m2e.core.embedder.IMaven; import org.eclipse.m2e.core.project.IMavenProjectFacade; import org.eclipse.m2e.core.ui.internal.editing.PomEdits; /** * @author mkleint */ public class XmlUtils { private static final Logger log = LoggerFactory.getLogger(XmlUtils.class); public static Element findChild(Element parent, String name) { return PomEdits.findChild(parent, name); } public static List<Element> findChilds(Element parent, String name) { return PomEdits.findChilds(parent, name); } public static String getTextValue(Node element) { return PomEdits.getTextValue(element); } /** * finds exactly one (first) occurence of child element with the given name (eg. dependency) that fulfills conditions * expressed by the Matchers (eg. groupId/artifactId match) * * @param parent * @param name * @param matchers * @return */ public static Element findChild(Element parent, String name, PomEdits.Matcher... matchers) { return PomEdits.findChild(parent, name, matchers); } /** * what is this method supposed to do? for the sourceViewer find the associated file on disk and for that one find the * IProject it belongs to. The required condition for the IProject instance is that project relative path of the file * shall only be pom.xml (thus no nested, unopened maven pom). So that when * MavenPlugin.getMavenProjectManager().getProject(prj); is called later on the instance, it actually returns the * maven model facade for the pom.xml backing the sourceViewer. * * @param sourceViewer * @return */ public static IProject extractProject(ITextViewer sourceViewer) { ITextFileBuffer buf = FileBuffers.getTextFileBufferManager().getTextFileBuffer(sourceViewer.getDocument()); if(buf == null) { //eg. for viewers of pom files in local repository return null; } IFileStore folder = buf.getFileStore(); File file = new File(folder.toURI()); IPath path = Path.fromOSString(file.getAbsolutePath()); Stack<IFile> stack = new Stack<IFile>(); //here we need to find the most inner project to the path. //we do so by shortening the path and remembering all the resources identified. // at the end we pick the last one from the stack. is there a catch to it? IFile ifile = ResourcesPlugin.getWorkspace().getRoot().getFileForLocation(path); if(ifile != null) { stack.push(ifile); } while(path.segmentCount() > 1) { IResource ires = ResourcesPlugin.getWorkspace().getRoot().findMember(path); if(ires != null && ires instanceof IFile) { stack.push((IFile) ires); } path = path.removeFirstSegments(1); } IFile res = stack.empty() ? null : stack.pop(); if(res != null) { IProject prj = res.getProject(); //the project returned is in a way unrelated to nested child poms that don't have an opened project, //in that case we pass along a wrong parent/aggregator if(res.getProjectRelativePath().segmentCount() != 1) { //if the project were the pom's project, the relative path would be just "pom.xml", if it's not just throw it out of the window.. prj = null; } return prj; } return null; } public static MavenProject extractMavenProject(ITextViewer sourceViewer) { //look in the sourceViewer's cache only if(sourceViewer instanceof IAdaptable) { return (MavenProject) ((IAdaptable) sourceViewer).getAdapter(MavenProject.class); } return null; } /** * converts an InputLocation to a file path on the local disk, null if not available. still the input source's model * value can be used further.. * * @param location * @return */ public static File fileForInputLocation(InputLocation location, MavenProject origin) { InputSource source = location.getSource(); if(source != null) { //MNGECLIPSE-2539 apparently if maven can't resolve the model from local storage, //the location will be empty. not only applicable to local repo models but //apparently also to models in workspace not reachable by relativePath String loc = source.getLocation(); File file = null; if(loc != null) { file = new File(loc); } else { //try to find pom by coordinates.. String modelId = source.getModelId(); if(origin.getModel().getId().equals(modelId) && origin.getFile() != null) { return origin.getFile(); } String[] splitStrings = modelId.split(":"); assert splitStrings.length == 3; IMavenProjectFacade facade = MavenPlugin.getMavenProjectRegistry().getMavenProject(splitStrings[0], splitStrings[1], splitStrings[2]); if(facade != null) { file = facade.getPomFile(); } else { //if not in the workspace, try looking into the local repository. IMaven maven = MavenPlugin.getMaven(); try { String path = maven.getArtifactPath(maven.getLocalRepository(), splitStrings[0], splitStrings[1], splitStrings[2], "pom", null); if(path != null) { file = new File(maven.getLocalRepositoryPath(), path); } } catch(CoreException e) { log.error("Failed to calculate local repository path of artifact", e); } } } return file; } return null; } /** * originally copied from org.eclipse.wst.xml.ui.internal.hyperlink.XMLHyperlinkDetector this method grabs the * IDOMModel for the IDocument, performs the passed operation on the node at the offset and then releases the * IDOMModel operation's Node value is also an instance of IndexedRegion * * @param offset */ public static void performOnCurrentElement(IDocument document, int offset, NodeOperation<Node> operation) { assert document != null; assert operation != null; // get the current node at the offset (returns either: element, // doctype, text) IStructuredModel sModel = null; try { sModel = StructuredModelManager.getModelManager().getExistingModelForRead(document); if(sModel != null) { IndexedRegion inode = sModel.getIndexedRegion(offset); if(inode == null) { inode = sModel.getIndexedRegion(offset - 1); } if(inode instanceof Node) { operation.process((Node) inode, sModel.getStructuredDocument()); } } } finally { if(sModel != null) { sModel.releaseFromRead(); } } } /** * this method grabs the IDOMModel for the IDocument, performs the passed operation on the root element of the * document and then releases the IDOMModel root Element value is also an instance of IndexedRegion * * @param doc * @param operation */ public static void performOnRootElement(IDocument doc, NodeOperation<Element> operation) { assert doc != null; assert operation != null; IDOMModel domModel = null; try { domModel = (IDOMModel) StructuredModelManager.getModelManager().getExistingModelForRead(doc); if(domModel == null) { throw new IllegalArgumentException("Document is not structured: " + doc); } IStructuredDocument document = domModel.getStructuredDocument(); Element root = domModel.getDocument().getDocumentElement(); operation.process(root, document); } finally { if(domModel != null) { domModel.releaseFromRead(); } } } public static void performOnRootElement(IFile resource, NodeOperation<Element> operation) throws IOException, CoreException { performOnRootElement(resource, operation, false); } public static void performOnRootElement(IFile resource, NodeOperation<Element> operation, boolean autoSave) throws IOException, CoreException { assert resource != null; assert operation != null; IDOMModel domModel = null; try { domModel = (IDOMModel) StructuredModelManager.getModelManager().getModelForRead(resource); if(domModel == null) { throw new IllegalArgumentException("Document is not structured: " + resource); } IStructuredDocument document = domModel.getStructuredDocument(); Element root = domModel.getDocument().getDocumentElement(); operation.process(root, document); if(autoSave && domModel.getReferenceCountForEdit() == 0) { domModel.save(); } } finally { if(domModel != null) { domModel.releaseFromRead(); } } } /* * calculates the path of the node up in the hierarchy, example of result is project/build/plugins/plugin * level parameter designates the number of parents to climb eg. for level 2 the result would be plugins/plugin * level -1 means all the way to the top. */ public static String pathUp(Node node, int level) { StringBuffer buf = new StringBuffer(); int current = level; while(node != null && current > 0) { if(node instanceof Element) { if(buf.length() > 0) { buf.insert(0, "/"); } buf.insert(0, node.getNodeName()); current = current - 1; } node = node.getParentNode(); } return buf.toString(); } }