/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 WARRANTIESOR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.aries.util.manifest;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.jar.Attributes;
import java.util.jar.Manifest;
import org.apache.aries.util.filesystem.IDirectory;
import org.apache.aries.util.filesystem.IFile;
import org.apache.aries.util.io.IOUtils;
/**
* This class contains utilities for parsing manifests. It provides methods to
* parse the manifest, read a manifest into a map and to split an manifest
* entry that follows the Import-Package syntax.
*/
public class ManifestProcessor
{
/**
* Reads a manifest's main attributes into a String->String map.
* <p>
* Will always return a map, empty if the manifest had no attributes.
*
* @param mf The manifest to read.
* @return Map of manifest main attributes.
*/
public static Map<String, String> readManifestIntoMap(Manifest mf){
HashMap<String, String> props = new HashMap<String, String>();
Attributes mainAttrs = mf.getMainAttributes();
if (mainAttrs!=null){
Set<Entry<Object, Object>> attributeSet = mainAttrs.entrySet();
if (attributeSet != null){
// Copy all the manifest headers across. The entry set should be a set of
// Name to String mappings, by calling String.valueOf we do the conversion
// to a string and we do not NPE.
for (Map.Entry<Object, Object> entry : attributeSet) {
props.put(String.valueOf(entry.getKey()), String.valueOf(entry.getValue()));
}
}
}
return props;
}
/**
* mapToManifest
*/
public static Manifest mapToManifest (Map<String,String> attributes)
{
Manifest man = new Manifest();
Attributes att = man.getMainAttributes();
att.putValue(Attributes.Name.MANIFEST_VERSION.toString(), Constants.MANIFEST_VERSION);
for (Map.Entry<String, String> entry : attributes.entrySet()) {
att.putValue(entry.getKey(), entry.getValue());
}
return man;
}
/**
* This method parses the manifest using a custom manifest parsing routine.
* This means that we can avoid the 76 byte line length in a manifest providing
* a better developer experience.
*
* @param in the input stream to read the manifest from.
* @return the parsed manifest
* @throws IOException
*/
public static Manifest parseManifest(InputStream in) throws IOException
{
Manifest man = new Manifest();
try
{
// I'm assuming that we use UTF-8 here, but the jar spec doesn't say.
BufferedReader reader = new BufferedReader(new InputStreamReader(in, "UTF-8"));
String line;
StringBuilder attribute = null;
String namedAttribute = null;
do {
line = reader.readLine();
// if we get a blank line skip to the next one
if (line != null && line.trim().length() == 0) continue;
if (line != null && line.charAt(0) == ' ' && attribute != null) {
// we have a continuation line, so add to the builder, ignoring the
// first character
attribute.append(line.trim());
} else if (attribute == null) {
attribute = new StringBuilder(line.trim());
} else if (attribute != null) {
// We have fully parsed an attribute
int index = attribute.indexOf(":");
String attributeName = attribute.substring(0, index).trim();
// TODO cope with index + 1 being after the end of attribute
String attributeValue = attribute.substring(index + 1).trim();
if ("Name".equals(attributeName)) {
man.getEntries().put(attributeValue, new Attributes());
namedAttribute = attributeValue;
} else {
Attributes.Name nameToAdd = new Attributes.Name(attributeName);
if (namedAttribute == null || !man.getMainAttributes().containsKey(nameToAdd)) {
man.getMainAttributes().put(nameToAdd, attributeValue);
} else {
man.getAttributes(namedAttribute).put(nameToAdd, attributeValue);
}
}
if (line != null) attribute = new StringBuilder(line.trim());
}
} while (line != null);
}
finally {
IOUtils.close(in);
}
return man;
}
/**
* Obtain a manifest from an IDirectory.
*
* @param appDir
* @param manifestName the name of manifest
* @return Manifest, or null if none found.
* @throws IOException
*/
public static Manifest obtainManifestFromAppDir(IDirectory appDir, String manifestName) throws IOException{
IFile manifestFile = appDir.getFile(manifestName);
Manifest man = null;
if (manifestFile != null) {
man = parseManifest(manifestFile.open());
}
return man;
}
/**
*
* Splits a delimiter separated string, tolerating presence of non separator commas
* within double quoted segments.
*
* Eg.
* com.ibm.ws.eba.helloWorldService;version="[1.0.0, 1.0.0]" &
* com.ibm.ws.eba.helloWorldService;version="1.0.0"
* com.ibm.ws.eba.helloWorld;version="2";bundle-version="[2,30)"
* com.acme.foo;weirdAttr="one;two;three";weirdDir:="1;2;3"
* @param value the value to be split
* @param delimiter the delimiter string such as ',' etc.
* @return List<String> the components of the split String in a list
*/
public static List<String> split(String value, String delimiter)
{
List<String> result = new ArrayList<String>();
if (value != null) {
String[] packages = value.split(delimiter);
for (int i = 0; i < packages.length; ) {
String tmp = packages[i++].trim();
// if there is a odd number of " in a string, we need to append
while (count(tmp, "\"") % 2 == 1) {
// check to see if we need to append the next package[i++]
tmp = tmp + delimiter + packages[i++].trim();
}
result.add(tmp);
}
}
return result;
}
/**
* count the number of characters in a string
* @param parent The string to be searched
* @param subString The substring to be found
* @return the number of occurrence of the subString
*/
private static int count(String parent, String subString) {
int count = 0 ;
int i = parent.indexOf(subString);
while (i > -1) {
if (parent.length() >= i+1)
parent = parent.substring(i+1);
count ++;
i = parent.indexOf(subString);
}
return count;
}
}