Skip to content
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 github version range support #1335

Closed
wants to merge 12 commits into from

Conversation

computmaxer
Copy link

I know this is not the proper way to submit contributions, but I want to get feedback from dart-lang folks and make sure it is something that could potentially be accepted before I spend any more time on this.

This is a fix for #1250. I work for Workiva (met some of you at the Dart Summit!) and we now have quite a few dart libraries in our Github organization. Some of these are public and OSS, but many of them cannot be. #1250 is a big issue for us for the private libraries as it requires us to update pubspecs all over the place anytime a patch version of some library is released, and consumers end up having a big list of dependency overrides while those transitions are in progress.

This PR implements version ranges for github dependencies with the following in a pubspec.yaml:

dependencies:
  w_flux:
    version: "^1.0.0"
    git: [email protected]:Workiva/w_flux.git

It assumes that versions will exist as git tags on the repo, and uses git tag -l to list the tags. It then collects a list of the tags that are considered valid version strings using Version.parse().

The biggest issue I have had so far is dealing with git tags that include the "v" in the name, i.e. "v1.0.0". I can strip the "v" before giving it to the version solver, but then when it tries to get the ref (_getRef) it of course cannot find it without the "v" that was previously removed. At Workiva we do not use the preceding "v" in our tag names, but it seems like many people do so I am assuming this will be a big issue for this feature.

It seems like it would be possible for the Version class to accept "v" versions strings in its parse constructor, and keep track of it so that it can put it back when toString() is called. This would require a change to pub_semver, though.

I'm hoping to get some feedback from the Dart team on whether or not we should continue down this path. Please let me know your thoughts!

@nex3 @kevmoo @munificent

@nex3
Copy link
Member

nex3 commented Sep 21, 2015

I don't think you need to do that many backflips to get "v" versions working. Just add logic to GitSource._getEffectiveRef to check for refs named "v$version" and "$version". That way you can have everything else just track the version number as part of the package ID, like it does for hosted dependencies.

@computmaxer computmaxer force-pushed the github-version-ranges branch 2 times, most recently from 10803f1 to 3be3412 Compare November 10, 2015 16:08
@computmaxer computmaxer force-pushed the github-version-ranges branch from 3be3412 to bb0fb5a Compare November 10, 2015 16:10
@computmaxer
Copy link
Author

Sorry for the delay! Finally had some time to spend on this yesterday. I've implemented @nex3 's suggestion for dealing with "v" version tags.

I also fixed a major issue with my initial approach. I was creating bare-bones Pubspec objects with just name and version in getVersions(), but it turns out those pubspecs need to be complete since one will eventually be used to determine which nested dependencies to get. So unfortunately we'll need to return the entire pubspec for each potential valid version. I have implemented that here by calling describeUncached() with each valid version string in a PackageId. This has the potential to impact performance since that will checkout each tag and load a pubspec, serially.

To help combat that issue I added a flag to enable this feature per dependency. Looks like this:

dependencies:
  w_flux:
    version: "^1.0.0"
    git: 
      url: [email protected]:Workiva/w_flux.git
      use_version_tags: true

Pub will only getVersions() for a particular git dependency if use_version_tags is true. This has the additional benefit of insulating this feature from existing tests and functionality, reducing its impact.


All unit tests are passing and I have tested this with a number of repositories, with and without "v"s in the version tags. Let me know what you think and what the next steps would be if you want to move forward with this solution.

@greglittlefield-wf
Copy link
Contributor

If checking out each tag is slow, would using git show (e.g., git show TAGNAME:pubspec.yaml) instead of git checkout help to significantly reduce that performance hit?

@computmaxer
Copy link
Author

@greglittlefield-wf nice! Didn't know about git show. I've added a commit that uses that instead of describeUncached().

@kevmoo
Copy link
Member

kevmoo commented Nov 10, 2015

@computmaxer If you're fixing your pull request, please rebase against master and do a --force push to your staging branch. This makes the pull request "clean".

} on FormatException {}
}

if (validVersions.length > 0) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

use .isNotEmpty

@computmaxer
Copy link
Author

@kevmoo comments addressed. Would you like all commits squashed into one as well?

var ref = _getEffectiveRef(id);
return git
.run(["rev-list", "--max-count=1", ref], workingDir: _repoCachePath(id))
.then((result) => result.first)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

By async, I also meant getting red of .then and .catchError and using await, too. 😄

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok. Do you guys always prefer async/await over future chaining, then?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In general, yes. It's easier to read and our debugger actually has special support for async/await – that we don't get if you use the Future API directly.

@nex3 has found a few cases where it's better to use the API directly, but she'd have to comment.

@kevmoo
Copy link
Member

kevmoo commented Nov 10, 2015

@computmaxer I'd wait to squash until reviews are done – but it's good to do once you think you're done

- Use .isNotEmpty instead of .length > 0
- Use async/await instead of Future chaining
@computmaxer computmaxer force-pushed the github-version-ranges branch from 9409905 to 93213d1 Compare November 10, 2015 19:26
workingDir: _repoCachePath(id));
return result.first;
}
throw e;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Use rethrow – it preserves the stack trace

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nice 👍

@travissanderson-wf
Copy link

@kevmoo @nex3 I think this is ready for your eyeballs again, thanks!

/// Anything that Version.parse understands is considered a version,
/// it will also attempt to strip off a preceding 'v' e.g. v1.0.0
Future<List<Pubspec>> getVersions(String name, description) async {
if (!_useVersionTags(description)) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm really not a fan of requiring users to explicitly annotate to get this behavior. For the most part, I don't expect people to be using version declarations for Git dependencies at all unless they want some sort of resolution, so requiring them to write more stuff seems annoying. And now that we're using git show, it shouldn't be too bad to get all the pubspecs.

On the other hand, we should short-circuit if the description doesn't have a version listed.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The issue I'm having here is that I can't seem to access the version in this method, as it is not part of the description.

w_transport:
  version: ^1.0.0
  git: [email protected]:Worvia/w_transport.git

description --> "[email protected]:Workiva/w_transport.git"

w_transport:
  version: ^1.0.0
  git: 
    url: [email protected]:Worvia/w_transport.git
    use_version_tags: true

description --> {'url': '[email protected]:Worvia/w_transport.git', 'use_version_tags': true}

Any idea how to access version so I can short-circuit based on it?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It'd be possible to access version if it was in the description under git, like you initially had it, right?

dependencies:
  w_flux:
    version: "^1.0.0"
    git: [email protected]:Workiva/w_flux.git

Is there any reason that version can't be under git?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The version solver (which is abstracted from the package source) won't respect it when it is under git. So you'd have include the same version entry twice.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yikes, good point. I'd just change Source.getVersions() to take a PackageRef rather than a name and description, and check whether the version constraint is any or not.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you mean PackageDep instead of PackageRef? PackageRef does not have a VersionConstraint available.

@travissanderson-wf
Copy link

+1, our team is currently in dependency_overrides hell to work around the issue, we'd definitely be happy to add a flag to every dependency for this solution

@kevmoo
Copy link
Member

kevmoo commented Nov 24, 2015

@travissanderson-wf @trentgrover-wf I'll chat with @nex3 to see what our options are – reading #1327, it sounds like you'd be open to running your own server

@computmaxer
Copy link
Author

@kevmoo we are open to it, but would rather have support for Git version solving to avoid maintenance headaches and the auth hurdle.

@nex3:

I don't think we're going to be able to trigger this based on a version constraint existing; that information just doesn't get piped where it would need to go, and it doesn't make sense in general to add that piping.

Why doesn't it make sense to add that piping? It seems like getVersions would just need to be given a PackageDep instead of a PackageRef. Looking briefly at the code, it seems like that might not be a trivial change (may affect a lot of the version solver code), but wouldn't it make sense to have constraint information when solving the version? How does that become a blocker for this feature? Pub already has the design/architecture to support version solving across different package sources, but only hosted packages are actually able to use it. Doesn't it make sense to refactor that functionality slightly to allow the design to be utilized?

@trentgrover-wf
Copy link

@kevmoo any further thoughts on this issue?

@travissanderson-wf
Copy link

@kevmoo @nex3 I already gave @computmaxer and his team breakfast pizza to thank him for getting this fixed for us, as you know that is a morally binding contract so he is feeling a lot of heat right now since we still can't use version ranges on github dependencies :)

@nex3
Copy link
Member

nex3 commented Dec 9, 2015

If you're looking to only allow authenticated downloads, I think your best bet is to either run some sort of VPN or to use HTTP basic auth and have people include their credentials in the PUB_HOSTED_URL variable. Either way you'll also need to set up your server to fall back on pub.dartlang.org packages.

@trentgrover-wf
Copy link

So is there no chance of a GitHub based solution (eliminates a bunch of infrastructure requirements)?

If you could point us to your specific areas of concern, we're more than happy to devote some dev time to it, but it's sounding like the NO is hardening...

@nex3
Copy link
Member

nex3 commented Dec 14, 2015

I don't think we're going to find a way to make version range resolution work with GitHub, no.

@johncblandii
Copy link

That kind of sucks to go through so much PR review/work to say "no" to the whole idea of it later.

The feature is useful. Hope you reconsider @nex3.

@kevmoo
Copy link
Member

kevmoo commented Dec 14, 2015

@johncblandii It absolutely sucks. We're very sorry.

We try very hard to make sure these types of changes don't have long-term implications or side-effects that can hurt the ecosystem.

@nex3 is especially diligent when it comes to these types of changes – although, as she mentioned, we realized there were gnarly edge cases we didn't initially consider when we first encouraged the work to proceed.

We'd be more than happy to discuss other approaches to this problem – we know path dependencies are a pain.

@travissanderson-wf
Copy link

Describing it as a pain is a bit of an understatement. We have a number of in-house libraries written in Dart and a handful of apps consuming those libraries. Our options today are either move everything in lockstep (dependencies of dependencies included) or to use a horrific set of dependency_overrides. The other prescribed option of running a private server with VPN access required puts an unnecessary burden of access control and infrastructure on the end-users of the Dart language. None of the other services we use require restricted network access, we do not generally use a VPN and can work with all our other "cloud" services from anywhere.

It's a little demoralizing as a company that has evangelized Dart and is offering to even do the work ourselves. I'm sure we're not alone with this being a major difficulty for anyone else trying to use Dart in a medium-large scale company. We would be elated to even use the "cumbersome" flag on every dependency. Our team alone has 40 lines (not exaggerating) of dependency_overrides entries in our primary app - it's almost a mirror of the dependencies section.

@johnryan-wf
Copy link

In hindsight, knowing this was a dead end would have allowed us to work on an internal pub server.

We want to make sure that whatever solution we end up with is solid, but I'm not sure why this can't be accomplished (in principle.) It seems like an elegant way to support private dependencies.

@kevmoo
Copy link
Member

kevmoo commented Dec 15, 2015

I'll chat w/ @nex3 a bit more tomorrow. We know git (and GitHub) is a really close fit here.

Maybe we can get creative.

@travissanderson-wf
Copy link

Thanks, @kevmoo @nex3. For what it's worth, if you identify some work you need done we'd be happy to provide some bandwidth.

@trentgrover-wf
Copy link

@kevmoo @nex3 any further update? sorry to badger you

@munificent
Copy link
Member

Various holidays and related vacations are kicking in, so @kevmoo and @nex3 might be offline for a while, but they will follow up on this. :)

@trentgrover-wf
Copy link

👍

@travissanderson-wf
Copy link

Have a good holiday vacation, all! See you when you get back.

darty-bird-xmas

@trentgrover-wf
Copy link

Now that the holidays have passed, any further update @kevmoo @nex3? We're trying to nail down how we want to address this issue internally, so your answer will help us determine what resources to devote to it.

@trentgrover-wf
Copy link

For posterity, here's where we landed:

  • Google will not be merging GitHub version range support because of the issues noted above and because the risk associated with introducing breaking changes this low in the stack is too high.
  • We're going with the private pub server route:
    • Google has committed to adding some oauth capability to pub so that pub get can be performed in a secure way without requiring the auth credentials to reside in the pubspec (as would be required if using only HTTP basic auth). Delivery date TBD.
    • Once we have a private pub server up and running with oauth, we'll use the hosted package syntax with a custom hosted url specified for each of our private pubspec dependencies.

@munificent
Copy link
Member

Sounds right to me. Just for the record (and you probably already know this):

  • Once we have a private pub server up and running with oauth, we'll use the hosted package syntax with a custom hosted url specified for each of our private pubspec dependencies.

This is supported already in pub, so there's no work on our end to do. It looks like:

dependencies:
  transmogrify:
    hosted:
      name: transmogrify
      url: http://your-package-server.com
    version: ^1.4.0

@computmaxer
Copy link
Author

@kevmoo @munificent @nex3
We planned on firewalling off our private pub server from the internet and only allowing our office/VPN IPs until the OAuth2 support for pub get is added.

I am having trouble getting this working with the Managed VM GAE deployment. I've added the firewall rule to allow our office IPs in Networking --> Firewall rules, but I can't seem to block access to everything else. I believe this would work if it was just a GCE deployment, but it seems GAE must add some rules or routes to allow traffic to be routed from the internet to the instances.

Do you guys know if it is possible to implement an IP whitelist for a GAE Managed VMs deployment?

@nex3
Copy link
Member

nex3 commented Feb 5, 2016

I'm far from an expert on AppEngine infrastructure, and I think the same is true for Kevin and Bob. Maybe @mkustermann would know; he works on pub.dartlang.org.

@kevmoo
Copy link
Member

kevmoo commented Feb 5, 2016

Try https://groups.google.com/forum/#!forum/app-engine-managed-vms

This is purely outside of the Dart team

@computmaxer
Copy link
Author

@nex3 @kevmoo
Do you all have any updates on the additional OAuth2 functionality for pub? We are yet to have a functioning private pub server approved for production use (I'll spare you the details 😄), and since we are approaching the end of Q1 I figured I'd check in to see where this is at. Thanks!

@nex3
Copy link
Member

nex3 commented Mar 15, 2016

I haven't had a chance to work on this yet. I'll try to get it in the works next quarter.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

9 participants