/* * Copyright 2013-2016 consulo.io * * 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 consulo.vfs.impl.archive; import com.intellij.openapi.components.ApplicationComponent; import com.intellij.openapi.fileTypes.FileType; import com.intellij.openapi.fileTypes.FileTypeRegistry; import com.intellij.openapi.util.SystemInfo; import com.intellij.openapi.util.io.FileUtil; import com.intellij.openapi.util.text.StringUtil; import com.intellij.openapi.vfs.VirtualFile; import com.intellij.openapi.vfs.impl.jar.JarHandler; import com.intellij.openapi.vfs.newvfs.ArchiveFileSystem; import com.intellij.openapi.vfs.newvfs.VfsImplUtil; import com.intellij.util.SystemProperties; import com.intellij.util.containers.ContainerUtil; import com.intellij.util.io.URLUtil; import consulo.fileTypes.ArchiveFileType; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import java.io.File; import java.io.IOException; import java.util.Set; /** * @author VISTALL * @since 06-Aug-16 * <p> * Base class for archive file types */ public abstract class ArchiveFileSystemBase extends ArchiveFileSystem implements consulo.vfs.ArchiveFileSystem, ApplicationComponent { private final Set<String> myNoCopyJarPaths; private final String myProtocol; protected ArchiveFileSystemBase(@NotNull String protocol) { myProtocol = protocol; boolean noCopy = SystemProperties.getBooleanProperty("idea.jars.nocopy", !SystemInfo.isWindows); myNoCopyJarPaths = noCopy ? null : ContainerUtil.newConcurrentSet(FileUtil.PATH_HASHING_STRATEGY); } public boolean isMyFile(@NotNull VirtualFile local) { FileType fileType = FileTypeRegistry.getInstance().getFileTypeByFileName(local.getName()); return fileType instanceof ArchiveFileType && ((ArchiveFileType)fileType).getFileSystem() == this; } @NotNull public abstract ArchiveFile createArchiveFile(@NotNull String filePath) throws IOException; @NotNull @Override public final String getProtocol() { return myProtocol; } @Override public void initComponent() { } @NotNull @Override public String getComponentName() { return getClass().getSimpleName(); } @Override public void disposeComponent() { } @Override public boolean isMakeCopyOfJar(@NotNull File originalJar) { return !(myNoCopyJarPaths == null || myNoCopyJarPaths.contains(originalJar.getPath())); } @Override public void setNoCopyJarForPath(String pathInJar) { if (myNoCopyJarPaths == null || pathInJar == null) return; int index = pathInJar.indexOf(URLUtil.ARCHIVE_SEPARATOR); if (index < 0) return; String path = FileUtil.toSystemIndependentName(pathInJar.substring(0, index)); myNoCopyJarPaths.add(path); } @NotNull @Override public String extractPresentableUrl(@NotNull String path) { return super.extractPresentableUrl(StringUtil.trimEnd(path, URLUtil.ARCHIVE_SEPARATOR)); } @Override protected String normalize(@NotNull String path) { final int jarSeparatorIndex = path.indexOf(URLUtil.ARCHIVE_SEPARATOR); if (jarSeparatorIndex > 0) { final String root = path.substring(0, jarSeparatorIndex); return FileUtil.normalize(root) + path.substring(jarSeparatorIndex); } return super.normalize(path); } @NotNull @Override protected String extractRootPath(@NotNull String path) { final int jarSeparatorIndex = path.indexOf(URLUtil.ARCHIVE_SEPARATOR); assert jarSeparatorIndex >= 0 : "Path passed to ArchiveFileSystem must have archive separator '!/': " + path; return path.substring(0, jarSeparatorIndex + URLUtil.ARCHIVE_SEPARATOR.length()); } @NotNull @Override protected String extractLocalPath(@NotNull String rootPath) { return StringUtil.trimEnd(rootPath, URLUtil.ARCHIVE_SEPARATOR); } @NotNull @Override protected String composeRootPath(@NotNull String localPath) { return localPath + URLUtil.ARCHIVE_SEPARATOR; } @NotNull @Override protected JarHandler getHandler(@NotNull VirtualFile entryFile) { return VfsImplUtil.getHandler(this, entryFile, JarHandler::new); } @Override public VirtualFile findFileByPath(@NotNull String path) { return VfsImplUtil.findFileByPath(this, path); } @Override public VirtualFile findFileByPathIfCached(@NotNull String path) { return VfsImplUtil.findFileByPathIfCached(this, path); } @Override public VirtualFile refreshAndFindFileByPath(@NotNull String path) { return VfsImplUtil.refreshAndFindFileByPath(this, path); } @Override public void refresh(boolean asynchronous) { VfsImplUtil.refresh(this, asynchronous); } @Nullable @Override public VirtualFile getLocalVirtualFileFor(@Nullable VirtualFile entryVFile) { return getVirtualFileForJar(entryVFile); } @Nullable @Override public VirtualFile findLocalVirtualFileByPath(@NotNull String path) { if (!path.contains(URLUtil.ARCHIVE_SEPARATOR)) { path += URLUtil.ARCHIVE_SEPARATOR; } return findFileByPath(path); } @Nullable public VirtualFile getVirtualFileForJar(@Nullable VirtualFile entryFile) { return entryFile == null ? null : getLocalByEntry(entryFile); } @Nullable public VirtualFile getJarRootForLocalFile(@NotNull VirtualFile file) { return getRootByLocal(file); } }