-
-
Notifications
You must be signed in to change notification settings - Fork 232
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add flag --submodules to add git submodules in the project folder as available packages. #1780
Add flag --submodules to add git submodules in the project folder as available packages. #1780
Conversation
Thanks for your pull request and interest in making D better, @FeepingCreature! We are looking forward to reviewing it, and you should be hearing from a maintainer soon.
Please see CONTRIBUTING.md for more information. If you have addressed all reviews or aren't sure how to proceed, don't hesitate to ping us with a simple comment. |
Note: I am not sure how the change in git version detection plays out in practice. I fixed it for submodules, but I don't have a testcase for the original usecase that motivated it. Would appreciate testing. |
Hm, CI doesn't seem to link in the new file..? edit: oops |
9e2ea0b
to
5b103bc
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Tests are a bare minimum.
Not sure how. Do I actually add a submodule in the test folder? edit: Or do I check out some test repo in the relevant test sh? |
I think the reason it's failing is it's now using the dub git version for the version of the test folders. Hm. Need to redesign a bit. |
461b219
to
f3b0e19
Compare
Okay, now it uses the Let's see if it passes, then we can think about testing. |
@wilzbach Okay, how do I test this? Only way I can think of is initialize a git repo, initialize a second git repo, make a commit, make a tag, add the second git repo as a submodule to the first, and see if dub --submodules picks it up. Does that sound good? It seems kind of overkill... |
b652afe
to
3fc1c76
Compare
Added a test that does just that. Seems to pass..? edit: Seems to fail. But why? |
31ea0fe
to
f84058c
Compare
Tests are green now. edit: The problem was that |
Is there any reason why this needs a flag that is manually run? |
It can absolutely be done automatically, but I didn't want to opt the users in to a new feature, especially one that changes the semantics of package lookup, without requiring some affirmative action. My intention was to set An alternative would be to set it as a flag in the |
Ping @wilzbach |
Suggestions for a later PR: Should dub consider all tags in the submodule as available versions? Should dub upgrade change the submodule's checked out tag? |
Is this really necessary ? From what I understand (the description is rather short), this will only affect the user's machine, which means anyone checking out the project will need to run this command, right ? We are using submodules for our project, and the best way we could come up with was just to use path dependency, because users of So what problem is this solving that path dependencies is not solving ? |
We specifically want to use dub as a dependency management tool. That is, we often have submodules that have dependencies that can clash with our own dependencies. The alternative would be to give up submodules, and they're just too convenient. The usecase is "there's a deep graph of repos, and they are all semver versioned, but they're all maintained inhouse and often updated together, and their updates sometimes break compatibility." Because they're maintained by the same group of people, it's usually the case that you want to test a library upgrade in the context it'll be used. Hence submodules. With external dub libraries I've sometimes done hacks like just copying my changes over the ~/.dub folder just to test if it does what I need. The point here is to not need to do things like that, but also be able to use Jenkins (hence local overrides being insufficient), and also not having to add and remove submodules during development (a workflow that git really does not like). |
So do you mix DUB dependencies and submodules ?
I suggest the opposite: Go all in on the submodules. This is what we do: Once you work past the initial pain (that is, realize you need some dummy dependencies because |
The problem is recursive submodules. Ie. A -> X1, A -> B -> X2, and enforcing that X1 and X2 are semver compatible. Just to elucidate the problem, one of our internal utility libraries is on version 19. (That's semver major. Yes that's 19 generations of breaking changes.)
Right now, yes. Of course, right now we build with an internal buildsystem that uses dub to build each dependency individually, so there's no version enforcement at all. That's why we want to switch to dub. |
So do you do recursive checkout of submodules ? What we do (and something we took from Sociomantic) is that we never, ever do recursive submodules. If App depends on Lib2 and Lib1, and Lib2 depends on Lib1, we just put them in
That doesn't make too much sense for me: DUB doesn't have to select a version, since the version are already selected (and fixed). At least when it comes to submodules.
IMO that's your real problem. If you use both systems, they have to be aware of each other, which is a pain, and will never be perfect. |
Not for building; every recursive dependency ends up in the root module. This is usually not a problem since the library dependencies tend to share the same five or so utility libraries.
See my comment about allowing dub to check out a different version of a submodule. But no, for now I'd be happy if it'd error in case of incompatible versions.
This is why we want to transition to dub only - and hence, this PR. |
…available packages. Adjust git version determination to support submodule folders
…ead of absolute --git-dir. With absolute --git-dir, we are unable to find the .gitmodules.
3b16e2a
to
b590ef7
Compare
@wilzbach I know you have other priorities, but then could you please resolve your change request for tests now that there's been a test for over two weeks? |
Ping. |
Can somebody please give me a pass/fail/change request on this? This PR is holding up work (specifically, our internal dub transition). |
@thewilsonator Thank you! |
@thewilsonator: there was clearly NO consensus on merging this flag with multiple contributors being unsure about the usefulness of this flag. What was your motivation for merging this? |
To be quite fair, there was clearly no functioning mechanism for arriving at a consensus either. This PR was open for two months. Over this time, the following pattern emerged: somebody comes by and leaves unsure feedback, I answer the feedback, and the person never shows up again. What exactly was I supposed to be doing here? edit: Regarding the flag: yes, it has limited usefulness. It's tied to our internal workflow with submodules. I don't know if there's another company that does submodules, because nobody talks about their internal workflows, here or otherwise. However, it may be instructive that at least one company decided that they're literally better off starting from scratch rather than working with dub, and frankly, PR workflows like this one may be why. Why submodules and not a package repository? Because in microservice development, dependencies usually change in pairs: upstream and downstream, at once. This cannot be tested in a tagged, semver-release-based workflow; you have to be able to commit an incremental state. That means we either need a way to set local overrides pointing to git commits in another repo, or submodules - in other words, either submodules or fake submodules manually. edit: Why work with upstream dub at all? Pretty much one reason only: IDE integration requires dub. Sure we could fork as funkwerk-dub and hope that all IDE plugins allow swapping out the dub binary, but that's also kind of a dissatisfactory workflow. The simple fact is, dub has no good answer for incremental development in internal ecosystems. (And yes, I am aware of local overrides, see above re Jenkins.) The point of this PR is to give it one, or at least the start of one. |
Yes, but that doesn't NOT mean that Nicholas can just arbitrarily merge things. Especially without providing any motivation or reasoning!
Yes, but it doesn't mean that this PR can be merged without motivation / reasoning.
You should have pinged someone with actual authority and time to approve things -> @atilaneves.
My review was a FAIL (!)
How so?
Which uses an existing feature and is more general-purpose as it doesn't solely apply to git submodules. |
No it was not! Your review was a request for a specific enhancement, ie. a test. The test was added.
Sure it can. You can have packages that are both on the registry and provided as submodules, ie. providing
I think that mixes two things: accepted versions and provided versions. Path-based dependencies will break once the package is used as a submodule in another package. |
Why? |
I mean, I guess you could ignore the path? Not sure if dub actually does this right now? The point is that if you have two submodules that both use the same transitive dependency, you need some way to advertise this dependency in the root package that uses both of them, or in any case get them to use the same version. So dub would have to ignore edit: Wanna take this to dlang slack? This seems like the sort of debate that's not well suited for an asynchronous comments-based exchange. |
BTW I do understand the problem, but there's nothing I can do about (as you have noticed my time is very limited). Maybe a NG discussion is a good start?
Are we talking about this?
https://github.com/vibe-d/vibe.d/blob/master/web/dub.sdl
I do think that the mindset of "this feature could work, but it's broken for this specific use-case. Hence, let's add a new feature" doesn't not scale long-term |
Yes, exactly.
I don't believe that subpackages can have their own versions.
I mean, I could implement that, but at this point I'm expecting that some people give feedback, the PR will lie dead for months of insistent pings, then somebody click merge and you'll revert it. My only hope is that you'll be more reluctant to revert a PR that you like. If the D foundation apparently isn't gonna designate someone, could I just straight up pay you money to take on reviewer duties for that PR? I'm getting a bit frustrated here. |
Update: I remember what the problem with edit: Though I guess that is mostly an extension of |
|
Hang on, |
Action plan for submodule supportGoing by feedback, the path most likely to have success is augmenting the
@wilzbach Does this seem likely to have success? |
What's deficient with adding the submodule as a path package in the package description? |
edit: Also I am not sure how dub handles overriding transitive submodule dependencies, or the case where the path of a transitive dependency doesn't exist, ie. non-recursive submodule checkout. But I'm willing to bet it's broken too. (two hours later: yep, it's very, very broken) edit:
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Stopping here - this patch seems much more complicated than it needs to be. As far as I can see, Git provides facilities to query information which this patch requires to be either explicitly specified in the configuration or clumsily attempts to guess it through string concatenation.
More importantly, it seems to be based on a bad assumption:
(Submodules are folders that have distinct git state (tags etc.) but do not contain a .git folder, since they share the .git folder of the parent project.)
This is only true for absorbed submodules. Newer versions of Git absorb submodules' git dirs automatically but this is not required. (Personally I prefer to keep them unabsorbed as that makes going out to edit .git/config
or .git/info/exclude
simpler.)
const submoduleInfo = execute([ | ||
"git", | ||
"-C", rootPath.toNativeString, | ||
"--git-dir=" ~ (rootScmPath.relativeTo(rootPath)).toNativeString, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why is --git-dir
necessary here? It was necessary in conjunction with --work-tree
before -C
was introduced.
|
||
foreach (line; submoduleInfo.output.lines) { | ||
const parts = line.split(" ").map!strip.filter!(a => !a.empty).array; | ||
const subPath = rootPath ~ parts[1]; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Bash code to enumerate registered active submodules:
git config -lz | \
while IFS=$'\n' read -r -d '' name value
do
if [[ "$name" =~ ^submodule\.(.*)\.url$ || ( "$name" =~ ^submodule\.(.*)\.active$ && "$value" == true ) ]]
then
printf '%s\0' "${BASH_REMATCH[1]}"
fi
done | \
sort -uz | \
mapfile -d '' -t list
See https://git-scm.com/docs/gitsubmodules#_active_submodules for reference.
To get the submodule git dir (of an absorbed module):
git rev-parse --git-path modules/"$m"
where $m
is the submodule name as per above list.
To get the submodule work dir:
git -C "$md" rev-parse --show-toplevel
where md
is the git dir above.
@@ -75,20 +75,24 @@ class Package { | |||
root = The directory in which the package resides (if any). | |||
parent = Reference to the parent package, if the new package is a | |||
sub package. | |||
scm_path = The directory in which the VCS (Git) stores its state. | |||
Different than root/.git for submodules. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This seems unnecessary
this(Json json_recipe, NativePath root = NativePath(), Package parent = null, string version_override = "") | ||
this(Json json_recipe, NativePath root = NativePath(), Package parent = null, | ||
NativePath scm_path = NativePath(), string version_override = "") | ||
{ | ||
import dub.recipe.json; | ||
|
||
PackageRecipe recipe; | ||
parseJson(recipe, json_recipe, parent ? parent.name : null); | ||
this(recipe, root, parent, version_override); | ||
this(recipe, root, parent, version_override, scm_path); | ||
} | ||
/// ditto | ||
this(PackageRecipe recipe, NativePath root = NativePath(), Package parent = null, string version_override = "") | ||
this(PackageRecipe recipe, NativePath root = NativePath(), Package parent = null, | ||
string version_override = "", NativePath scm_path = NativePath()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ditto
try recipe.version_ = determineVersionFromSCM(root); | ||
try recipe.version_ = determineVersionFromSCM(root, scm_path); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ditto
scm_path = The directory in which the VCS (Git) stores its state. | ||
Different than root/.git for submodules! |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ditto
static Package load(NativePath root, NativePath recipe_file = NativePath.init, Package parent = null, string version_override = "") | ||
static Package load(NativePath root, | ||
NativePath recipe_file = NativePath.init, Package parent = null, | ||
string version_override = "", NativePath scm_path = NativePath.init) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ditto
auto ret = new Package(recipe, root, parent, version_override); | ||
auto ret = new Package(recipe, root, parent, version_override, scm_path); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ditto
if (scm_path.empty) { | ||
scm_path = path ~ ".git"; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Don't do this
auto hpath = (path ~ ".git/HEAD").toNativeString(); | ||
auto hpath = (scm_path ~ "HEAD").toNativeString(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This was bad code and this is not an improvement :)
Also, adjust git version determination to support submodule folders with the -C flag.
(Submodules are folders that have distinct git state (tags etc.) but do not contain a
.git
folder, since they share the.git
folder of the parent project.)