This page is not created by, affiliated with, or supported by Slack Technologies, Inc.
2023-04-01
Channels
- # announcements (11)
- # babashka (21)
- # beginners (57)
- # biff (5)
- # calva (75)
- # clj-commons (20)
- # cljs-dev (27)
- # clojars (88)
- # clojure (18)
- # clojure-europe (11)
- # clojure-germany (1)
- # clojurescript (52)
- # datalevin (25)
- # emacs (2)
- # graphql (2)
- # gratitude (4)
- # off-topic (6)
- # pathom (16)
- # reagent (3)
- # releases (4)
- # shadow-cljs (7)
- # spacemacs (9)
- # transit (3)
- # xtdb (5)
Are there any maven experts here? I've been writing my own pom parser + dependency resolver and the full flow of how POMs are resolved is still super wacky to me and idk if I get it (details in thread)
Then there is a step where the inheritance tree is collapsed and properties are subbed
/**
* POM (xml file)
* |
* | Parse XML
* V
* PomInfo
* |
* | Fetch all parent poms
* V
* ChildHavingPomInfo
* |
* | Squash parent poms and replace properties
* V
* EffectivePomInfo
* |
* | Fetch BOMs and other deps with import scope
* v
* ?????
* |
* | Collapse dependency and dependency-management
* v
* PomManifest [ contains just list of deps ]
* |
* | Normalize snapshots and version ranges
* v
* PomManifest [ type unchanged ]
*/
and i've been reading both other code doing this same task and the documentation - its helping but its painful
There are a couple maven books they wrote a long while ago that might actually cover some of this
You didn’t mention property resolution above but that also should be in there
But yes, it is painful
this is the "flow" i got rn (huge code chunks)
record PomInfo(
PomGroupId groupId,
PomArtifactId artifactId,
PomVersion version,
List<PomDependency> dependencies,
PomParent parent,
List<PomDependency> dependencyManagement,
List<PomProperty> properties,
PomPackaging packaging
) {
}
|
v
record ChildHavingPomInfo(
PomGroupId groupId,
PomArtifactId artifactId,
PomVersion version,
List<PomDependency> dependencies,
List<PomDependency> dependencyManagement,
List<PomProperty> properties,
PomPackaging packaging,
Optional<ChildHavingPomInfo> child
) {
|
v
record EffectivePomInfo(
PomGroupId groupId,
PomArtifactId artifactId,
PomVersion version,
List<PomDependency> dependencies,
List<PomDependency> dependencyManagement,
PomPackaging packaging
) {
static EffectivePomInfo from(final ChildHavingPomInfo childHavingPomInfo) {
var properties = new LinkedHashMap<String, String>();
var top = childHavingPomInfo;
while (top != null) {
for (PomProperty property : top.properties()) {
properties.put(property.key(), property.value());
}
top = top.child().orElse(null);
}
Function<String, String> resolve =
str -> resolveProperties(properties, str);
Function<PomDependency, PomDependency> resolveDep = dependency ->
new PomDependency(
dependency.groupId().map(resolve),
dependency.artifactId().map(resolve),
dependency.version().map(resolve),
dependency.exclusions().stream()
.map(exclusion -> new PomExclusion(
exclusion.groupId().map(resolve),
exclusion.artifactId().map(resolve)
))
.collect(Collectors.toUnmodifiableSet()),
dependency.type().map(resolve),
dependency.classifier().map(resolve),
dependency.optional().map(resolve),
dependency.scope().map(resolve)
);
PomGroupId groupId = PomGroupId.Undeclared.INSTANCE;
PomVersion version = PomVersion.Undeclared.INSTANCE;
PomPackaging packaging = PomPackaging.Undeclared.INSTANCE;
var dependencies = new LinkedHashMap<PomDependencyKey, PomDependency>();
var dependencyManagement = new LinkedHashMap<PomDependencyKey, PomDependency>();
top = childHavingPomInfo;
while (top != null) {
if (top.groupId() instanceof PomGroupId.Declared) {
groupId = top.groupId().map(resolve);
}
if (top.version() instanceof PomVersion.Declared) {
version = top.version().map(resolve);
}
if (top.packaging() instanceof PomPackaging.Declared) {
packaging = top.packaging().map(resolve);
}
top = top.child().orElse(null);
}
groupId.ifDeclared(value -> properties.put("project.groupId", value));
version.ifDeclared(value -> properties.put("project.version", value));
var artifactId = childHavingPomInfo.artifactId().map(resolve);
artifactId.ifDeclared(value -> properties.put("project.artifactId", value));
top = childHavingPomInfo;
while (top != null) {
top.dependencies()
.forEach(dependency -> {
var newDep = resolveDep.apply(dependency);
dependencies.put(PomDependencyKey.from(newDep), newDep);
});
top.dependencyManagement()
.forEach(dependency -> {
var newDep = resolveDep.apply(dependency);
dependencyManagement.put(PomDependencyKey.from(newDep), newDep);
});
top = top.child().orElse(null);
}
return new EffectivePomInfo(
groupId,
artifactId,
version,
dependencies.values().stream().toList(),
dependencyManagement.values().stream().toList(),
packaging
);
}
|
v
record PomManifest(
@Override List<Dependency> dependencies
) implements Manifest {
I'm curious why you're doing this :)
For what it is worth I find the maven APIs overwhelming too. I have learned a thing or two by reading https://github.com/clj-commons/pomegranate and https://github.com/clojure/tools.deps sources.
bright side - i have successfully detached resolution from all this nonsense in the same way tools.deps has
every time I read the Maven source I just want to throttle someone. it's such a paragon of 00's Java style, but it's all just data that could be maps and you would need none of the "abstractions" and "patterns"
I think maybe we've finally reached a point where new languages actually think about this early
I remember well building Java applications before Maven and repos existed
it was not fun
Trying to make sense of the Maven APIs put me in a similar mood to when I tried to make sense of Windows shell escaping rules.
yeah there is stuff maven does like "every repo is checked for every library" that feels insane
it's not only insane, it has all kinds of security issues
right now this is my conception of a coordiante
public interface Coordinate {
default Coordinate normalize(Library library, Cache cache) {
return this;
}
enum VersionComparison {
INCOMPARABLE,
GREATER_THAN,
LESS_THAN,
EQUAL_TO;
public static VersionComparison fromInt(int comparisonResult) {
return comparisonResult == 0 ? EQUAL_TO : comparisonResult > 0 ? GREATER_THAN : LESS_THAN;
}
}
CoordinateId id();
Manifest getManifest(Library library, Cache cache);
/**
* Gets the location of the given library on disk, assuming the library was located
* with this coordinate.
*
* <p>
* If the library is not downloaded on disk, this method will do so before returning.
* </p>
*/
Path getLibraryLocation(Library library, Cache cache);
default Optional<Path> getLibrarySourcesLocation(Library library, Cache cache) {
return Optional.empty();
}
default Optional<Path> getLibraryDocumentationLocation(Library library, Cache cache) {
return Optional.empty();
}
}
Manifest is just a named List<Dependency>
and thats what all this pom BS collapses into
public record MavenCoordinate(
Version version,
List<MavenRepository> repositories,
List<Scope> scopes
) implements Coordinate {
but the big ? for me at this level is whether exclusions are conceptually part of the coordinate
I would say yes
scopes tell you how to make different kinds of classpaths out of the same set of dependencies (at least that's how I think about it)
you seem to be missing classifiers too unless that's part of your version. classifiers are a weird little world of their own as they bring in conditional stuff
abstract InputStream getFile(
Library library,
Version version,
Classifier classifier,
Extension extension
) throws LibraryNotFound;
they are really important for a small percentage of Java libs
particularly those with native stuff
the two big conditionals they cover usually are jdk (which is rarely used anymore and shouldn't even be needed now that we have multi release jars), and native architecture
classifiers are weird in that all classifiers technically share the same pom and have the same coordinate. that they wedged docs and source into that is a bit of a hack
so i can conceive of how to add that to the MavenCoordinate - like use this classifier for the jar - but how would that affect resolution?
in deps I decided to make it part of the artifact: group/artifact[$classifier]
vs making it part of the version
which is already a little maven specific, but slapping Classifier on there feels gross...
classifier is really a variant of an artifact that is specific to jdk/arch/something else (we use "aot" as a classifier in a few of the contrib libs)
clojure itself publishes both the "normal" compiled jar, but also a "slim" classifier jar with just source
well like I said, those are imo hacks - they are also classifiers
really, Maven should have had a way to associate some kind of ancillary information (like Clojure metadata) to an artifact
you see why im thinking about this? I think tools.deps does its job for clojure, but would be neat if a foundational part of modern programming didn't elicit reactions like > I just want to throttle someone. and > a similar mood to when I tried to make sense of Windows shell escaping rules.
Fight the good fight :)
@U3JH98J4R are you writing your own version of maven and why? :)
@U04V15CAJ of the resolver, not the build tool
@U3JH98J4R and why, if I may ask?
Reasoning only extends as far as this https://clojurians.slack.com/archives/C0H28NMAS/p1680363571330679?thread_ts=1680362691.333169&cid=C0H28NMAS
Right, I'm asking since I considered doing something like this once when I had trouble compiling tools.deps.alpha (at the time) to native
It would be interesting to have a pure .clj solution (even more so if it had a chance of being adopted by tools.deps). The resolution mechanism isn't changing all the time is it?
I’d consider it if it got me everything I needed, but really this is just part of the puzzle
There’s so many features we can provide via the maven libs and it’s a lot to take all of that on
Which is why I’ve given up every time I’ve considered it :)
btw in the end I managed to compile tools.deps to native: https://github.com/babashka/tools-deps-native and this is one of the projects using it: https://github.com/babashka/tools.bbuild nothing more than a "look how far we can go" project at this point, in bb I'm still shelling out to Java via deps.clj to fetch deps