微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

如何将 Manifest 部分与 Gradle 和 shadowJar 合并

如何解决如何将 Manifest 部分与 Gradle 和 shadowJar 合并

我需要什么

我们使用 Gradle 和 shadowJar 打包我们的产品。我们使用的一些库利用 individual sections in Jar Manifests,特别是诸如 Implementation-Title 和 实施版本。这些有时会显示在我们的产品(输出)中,所以我希望它们能够在 shawdowJar-Process 中幸存下来。

示例

lib1.jar/meta-inf/MANIFEST.MF

Manifest-Version: 1.0
...

Name: org/some/lib
...
Implementation-Title: someLib
Implementation-Version: 2.3
...

lib2.jar/meta-inf/MANIFEST.MF

Manifest-Version: 1.0
...

Name: org/some/other/lib
...
Implementation-Title: someOtherLib
Implementation-Version: 5.7-RC
...

=> product.jar/meta-inf/MANIFEST.MF

Manifest-Version: 1.0
...

Name: org/some/lib
...
Implementation-Title: someLib
Implementation-Version: 2.3
...

Name: org/some/other/lib
...
Implementation-Title: someOtherLib
Implementation-Version: 5.7-RC
...

我发现了什么

project.shadowJar {
    manifest {
        attributes(["Implementation-Title" : "someLib"],"org/some/lib")
        attributes(["Implementation-Title" : "someOtherLib"],"org/some/other/lib")
    }
}

静态生成我想要的内容

project.shadowJar {
    manifest {
        for (dependency in includedDependencies) {
            // read in jar file and set attributes
        }
    }
}

Gradle 不高兴:“依赖项配置 ':project:products::compile' 已包含在依赖项解析中后,无法更改其依赖项。”

  • 当我定义一个新任务时
def dependencies = [];
project.tasks.register('resolveDependencies') {
    doFirst {
        gradleProject.configurations.compile.resolvedConfiguration.resolvedArtifacts.each {
            dependencies.add(it.file)
        }
    }
}
project.tasks['shadowJar'].dependsOn(project.tasks['resolveDependencies']);

project.shadowJar {
    manifest {
        // dependencies will be empty when this code is called
        for (dependency in dependencies) {
            // read in jar file and set attributes
        }
    }
}

依赖关系没有及时解决

我想知道的

如何在不影响 Gradle 的情况下访问依赖项?或者,是否有另一种方法将命名的各个部分与 shadowJar 合并?

解决方法

根据https://github.com/johnrengelman/shadow/issues/369应该使用shadowJar的Transformer接口来做到这一点。

所以来了:

import com.github.jengelman.gradle.plugins.shadow.transformers.Transformer;
import com.github.jengelman.gradle.plugins.shadow.transformers.TransformerContext;

import java.io.ByteArrayOutputStream;
import java.util.jar.Attributes;
import java.util.jar.Manifest;
import java.util.Map.Entry;

import shadow.org.apache.tools.zip.ZipOutputStream;
import shadow.org.apache.tools.zip.ZipEntry;
import shadow.org.codehaus.plexus.util.IOUtil;

import org.gradle.api.file.FileTreeElement;

import static java.nio.charset.StandardCharsets.*
import static java.util.jar.JarFile.*;

/**ManifestVersionMergeTransformer appends all version information sections from manifest files to the resulting manifest file.
  * @author Robert Lichtenberger
  */
public class ManifestMergeTransformer implements Transformer {

    String includePackages; // regular expression that must match a given package
    String excludePackages; // regular expression that must not match a given package

    private Manifest manifest;

    @Override
    public boolean canTransformResource(FileTreeElement element) {
        MANIFEST_NAME.equalsIgnoreCase(element.relativePath.pathString);
    }

    @Override
    public void transform(TransformerContext context) {
        if (manifest == null) {
            manifest = new Manifest(context.is);
        } else {
            Manifest toMerge = new Manifest(context.is);
            for (Entry<String,Attributes> entry : toMerge.getEntries().entrySet()) {
                if (mustInclude(entry.getKey())) {
                    manifest.getEntries().put(entry.getKey(),entry.getValue());
                }
            }
        }
        IOUtil.close(context.is);
    }

    private boolean mustInclude(String packageName) {
        return (includePackages == null || packageName.matches(includePackages)) && (excludePackages == null || !packageName.matches(excludePackages));
    }

    @Override
    public boolean hasTransformedResource() {
        return true;
    }

    @Override
    public void modifyOutputStream(ZipOutputStream os,boolean preserveFileTimestamps) {
        ZipEntry entry = new ZipEntry(MANIFEST_NAME);
        entry.time = TransformerContext.getEntryTimestamp(preserveFileTimestamps,entry.time);
        os.putNextEntry(entry);
        if (manifest != null) {
            ByteArrayOutputStream manifestContents = new ByteArrayOutputStream();
            manifest.write(manifestContents);
            os.write(manifestContents.toByteArray());
        }
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。