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

org.eclipse.tycho.versions.bundle.MutableBundleManifest Maven / Gradle / Ivy

There is a newer version: 3.0.5
Show newest version
/*******************************************************************************
 * Copyright (c) 2008, 2020 Sonatype Inc. and others.
 * This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License 2.0
 * which accompanies this distribution, and is available at
 * https://www.eclipse.org/legal/epl-2.0/
 *
 * SPDX-License-Identifier: EPL-2.0
 *
 * Contributors:
 *    Sonatype Inc. - initial API and implementation
 *    Nepomuk Seiler - set export-package attribute implementation
 *    Sebastien Arod - RequireBundle/FragmentHost/ImportPackage versions set/get
 *******************************************************************************/
package org.eclipse.tycho.versions.bundle;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.PushbackReader;
import java.io.Writer;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import org.eclipse.osgi.util.ManifestElement;
import org.eclipse.tycho.versions.engine.Versions;
import org.osgi.framework.BundleException;
import org.osgi.framework.Constants;

public class MutableBundleManifest {

    private final List attributes = new ArrayList<>();

    private String lineEnding = "";
    private String unparsed;

    public void add(ManifestAttribute attribute) {
        attributes.add(attribute);
    }

    public static MutableBundleManifest read(File file) throws IOException {
        try (InputStream is = new BufferedInputStream(new FileInputStream(file))) {
            return read(is);
        }
    }

    public static MutableBundleManifest read(InputStream is) throws IOException {
        PushbackReader br = new PushbackReader(new BufferedReader(new InputStreamReader(is, StandardCharsets.UTF_8)),
                1);

        MutableBundleManifest mf = new MutableBundleManifest();
        ManifestAttribute curr = null;

        String str;
        while ((str = readLineWithLineEnding(br, mf)) != null) {
            if (str.trim().isEmpty()) {
                break;
            } else if (str.charAt(0) == ' ') {
                if (curr == null) {
                    throw new IOException("");
                }
                curr.add(str);
            } else {
                curr = new ManifestAttribute(str);
                mf.add(curr);
            }
        }

        if (str != null) {
            StringBuilder sb = new StringBuilder(str);
            int ch;
            while ((ch = br.read()) != -1) {
                sb.append((char) ch);
            }
            mf.setUnparsed(sb.toString());
        }

        return mf;
    }

    private static String readLineWithLineEnding(PushbackReader reader, MutableBundleManifest mf) throws IOException {
        StringBuilder result = new StringBuilder();
        int ch, lastch = -1;

        while ((ch = reader.read()) != -1) {
            if (lastch == '\r') {
                if (ch == '\n') {
                    result.append((char) ch);
                    mf.setLineEndingWhenFirstLine("\r\n");
                } else {
                    reader.unread(ch);
                    mf.setLineEndingWhenFirstLine("\r");
                }
                break;
            }

            result.append((char) ch);

            if (ch == '\n' || ch == '\u2028' || ch == '\u2029' || ch == '\u0085') { // see Scanner#LINE_SEPARATOR_PATTERN
                mf.setLineEndingWhenFirstLine(new String(new char[] { (char) ch }));
                break;
            }

            lastch = ch;
        }

        if (result.length() > 0) {
            return result.toString();
        }
        return null;
    }

    private void setLineEndingWhenFirstLine(String lineEnding) {
        if (this.lineEnding.isEmpty() && lineEnding != null) {
            this.lineEnding = lineEnding;
        }
    }

    private void setUnparsed(String unparsed) {
        this.unparsed = unparsed;
    }

    public static void write(MutableBundleManifest mf, File file) throws IOException {
        try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
            write(mf, os);
        }
    }

    public static void write(MutableBundleManifest mf, OutputStream os) throws IOException {
        Writer w = new OutputStreamWriter(os, StandardCharsets.UTF_8);

        for (ManifestAttribute attribute : mf.attributes) {
            attribute.writeTo(w, mf.lineEnding);
        }

        if (mf.unparsed != null) {
            w.write(mf.unparsed);
        }

        w.flush();
    }

    public String getSymbolicName() {
        ManifestElement[] id = parseHeader(Constants.BUNDLE_SYMBOLICNAME);
        if (id != null && id.length > 0) {
            return id[0].getValue();
        }
        return null;
    }

    public String getVersion() {
        ManifestElement[] version = parseHeader(Constants.BUNDLE_VERSION);
        if (version != null && version.length > 0) {
            return version[0].getValue();
        }
        return null;
    }

    public String getFragmentHostSymbolicName() {
        ManifestElement[] fragmentHost = parseHeader(Constants.FRAGMENT_HOST);
        if (fragmentHost != null && fragmentHost.length > 0) {
            return fragmentHost[0].getValue();
        }
        return null;
    }

    public boolean isFragment() {
        return getFragmentHostSymbolicName() != null;
    }

    public String getFragmentHostVersion() {
        ManifestElement[] fragmentHost = parseHeader(Constants.FRAGMENT_HOST);
        if (fragmentHost != null && fragmentHost.length > 0) {
            return fragmentHost[0].getAttribute(Constants.BUNDLE_VERSION_ATTRIBUTE);
        }
        return null;
    }

    public void setFragmentHostVersion(String newVersion) {
        newVersion = Versions.toBaseVersion(newVersion);
        List fragmentHostElements = parseHeaderForMutation(Constants.FRAGMENT_HOST);
        if (fragmentHostElements == null || fragmentHostElements.isEmpty()) {
            return;
        }
        fragmentHostElements.get(0).setAttribute(Constants.BUNDLE_VERSION_ATTRIBUTE, newVersion);

        getAttribute(Constants.FRAGMENT_HOST).set(Constants.FRAGMENT_HOST, fragmentHostElements);
    }

    /**
     * Read the RequiredBundle with optional versions
     * 
     * @return
     */
    public Map getRequiredBundleVersions() {
        ManifestElement[] requireBundleElements = parseHeader(Constants.REQUIRE_BUNDLE);
        if (requireBundleElements == null || requireBundleElements.length == 0) {
            return Collections.emptyMap();
        }

        Map result = new HashMap<>();
        for (ManifestElement manifestElement : requireBundleElements) {
            String symbolicName = manifestElement.getValue();
            String versionRangeString = manifestElement.getAttribute(Constants.BUNDLE_VERSION_ATTRIBUTE);
            result.put(symbolicName, versionRangeString);
        }
        return result;
    }

    /**
     * Updates version range for bundles in Require-Bundle with the versions provided in the map.
     * 
     * @param versionsToUpdate
     *            the bundle/version-range map. Keys are the bundle symbolicnames to update and the
     *            values are the new version range to apply.
     */
    public void updateRequiredBundleVersions(Map versionsToUpdate) {
        List requireBundleElements = parseHeaderForMutation(Constants.REQUIRE_BUNDLE);
        if (requireBundleElements == null || requireBundleElements.isEmpty()) {
            return;
        }
        boolean somethingChanged = false;
        for (MutableManifestElement mutableManifestElement : requireBundleElements) {
            String bundleSymbolicName = mutableManifestElement.getValue();
            if (versionsToUpdate.containsKey(bundleSymbolicName)) {
                somethingChanged |= mutableManifestElement.setAttribute(Constants.BUNDLE_VERSION_ATTRIBUTE,
                        versionsToUpdate.get(bundleSymbolicName));
            }
        }
        if (somethingChanged) {
            getAttribute(Constants.REQUIRE_BUNDLE).set(Constants.REQUIRE_BUNDLE, requireBundleElements);
        }
    }

    /**
     * Get a map containing the name of packages in Import-Package manifest attribute as a map.
     * 

* The map keys are the package names and the values are the version range if present or null * when absent. * * @return */ public Map getImportPackagesVersions() { ManifestElement[] importPackageElements = parseHeader(Constants.IMPORT_PACKAGE); if (importPackageElements == null || importPackageElements.length == 0) { return Collections.emptyMap(); } Map result = new HashMap<>(); for (ManifestElement manifestElement : importPackageElements) { String packageName = manifestElement.getValue(); String versionRangeString = manifestElement.getAttribute(Constants.VERSION_ATTRIBUTE); result.put(packageName, versionRangeString); } return result; } /** * Updates version range for packages in Import-Package with the version provided in the map. * * @param versionsToUpdate * the package/version-range map. Keys are the package names to update and the values * are the new version range to apply. */ public void updateImportedPackageVersions(Map versionsToUpdate) { List importPackageElements = parseHeaderForMutation(Constants.IMPORT_PACKAGE); if (importPackageElements == null || importPackageElements.isEmpty()) { return; } boolean somethingChanged = false; for (MutableManifestElement mutableManifestElement : importPackageElements) { String packageName = mutableManifestElement.getValue(); if (versionsToUpdate.containsKey(packageName)) { somethingChanged |= mutableManifestElement.setAttribute(Constants.VERSION_ATTRIBUTE, versionsToUpdate.get(packageName)); } } if (somethingChanged) { getAttribute(Constants.IMPORT_PACKAGE).set(Constants.IMPORT_PACKAGE, importPackageElements); } } /** * Get a map containing the name of packages in Export-Package manifest attribute as a map. * * @return the package/version map. Keys are the package names and the values are the version if * present or null when absent. */ public Map getExportedPackagesVersion() { ManifestElement[] exportPackageElements = parseHeader(Constants.EXPORT_PACKAGE); if (exportPackageElements == null || exportPackageElements.length == 0) { return Collections.emptyMap(); } Map result = new HashMap<>(); for (ManifestElement manifestElement : exportPackageElements) { String packageName = manifestElement.getValue(); String versionRangeString = manifestElement.getAttribute(Constants.VERSION_ATTRIBUTE); result.put(packageName, versionRangeString); } return result; } /** * Updates version for packages in Export-Package with the version provided in the map. * * @param versionsToUpdate * the package/version map. Keys are the package names to update and the values are * the newVersion to apply. */ public void updateExportedPackageVersions(Map versionsToUpdate) { List exportPackageElements = parseHeaderForMutation(Constants.EXPORT_PACKAGE); if (exportPackageElements == null || exportPackageElements.isEmpty()) { return; } boolean somethingChanged = false; for (MutableManifestElement mutableManifestElement : exportPackageElements) { String packageName = mutableManifestElement.getValue(); if (versionsToUpdate.containsKey(packageName)) { somethingChanged |= mutableManifestElement.setAttribute(Constants.VERSION_ATTRIBUTE, versionsToUpdate.get(packageName)); } } if (somethingChanged) { getAttribute(Constants.EXPORT_PACKAGE).set(Constants.EXPORT_PACKAGE, exportPackageElements); } } private ManifestElement[] parseHeader(String name) { ManifestAttribute property = getAttribute(name); if (property == null) { return null; } try { return ManifestElement.parseHeader(name, property.getValue()); } catch (BundleException e) { throw new IllegalArgumentException("Could not parse bundle manifest", e); } } private List parseHeaderForMutation(String name) { ManifestAttribute property = getAttribute(name); if (property == null) { return null; } try { return MutableManifestElement.parseHeader(name, property.getValue()); } catch (BundleException e) { throw new IllegalArgumentException("Could not parse bundle manifest", e); } } private ManifestAttribute getAttribute(String name) { for (ManifestAttribute attribute : attributes) { if (attribute.hasName(name)) { return attribute; } } return null; } public void setVersion(String version) { ManifestAttribute attr = getAttribute(Constants.BUNDLE_VERSION); if (attr != null) { attr.set(Constants.BUNDLE_VERSION, version); } else { attributes.add(new ManifestAttribute(Constants.BUNDLE_VERSION, version)); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy