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 support for docker.runEnv properties and -D flags #938

Closed
wants to merge 1 commit into from

Conversation

stromnet
Copy link
Contributor

@stromnet stromnet commented Feb 5, 2018

Background:
When running in an environment which requires http_proxy to be set in a container, it's usually environment specific and not something to be specified in an image or the project's POM file.

Since there is already "specific" support for docker.buildArg through properties & -D flags, I thought it would not be too weird to add a docker.runEnv as well. Ideally I'd want to have all things configurable with property overrides (without requiring externalized config), but that is a much greater change I recon.

Example:
Instead of adding

<run>
  <env>
    <http_proxy>http://example.com:8080</http_proxy>
  </env
</run>

hardcoded in every project POM,

I can instead add

<properties>
  <docker.runEnv.http_proxy>http://example.com:8080</docker.runEnv.http_proxy>
</properties>

in either my .m2 settings.xml,

or, I could add as -D flag during build (for example in my default Jenkins template)

mvn .... -Ddocker.runEnv.http_proxy=http://example.com:8080

Change
This moves helper methods used for existing docker.buildArg into EnvUtil and uses the same when creating the ContainerCreateConfig in RunService.

Env vars for creating contains can now be set with docker.runEnv using
both Maven properties and -D flags, similar to how docker.buildArg is
supported.

Signed-off-by: Johan Ström <johan@stromnet.se>
@stromnet
Copy link
Contributor Author

stromnet commented Feb 5, 2018

Partially related: #386

@codecov
Copy link

codecov bot commented Feb 5, 2018

Codecov Report

Merging #938 into master will increase coverage by 0.42%.
The diff coverage is 66.66%.

@@             Coverage Diff              @@
##             master     #938      +/-   ##
============================================
+ Coverage     51.57%   51.99%   +0.42%     
- Complexity     1232     1270      +38     
============================================
  Files           144      144              
  Lines          7250     7301      +51     
  Branches        981      991      +10     
============================================
+ Hits           3739     3796      +57     
+ Misses         3195     3183      -12     
- Partials        316      322       +6
Impacted Files Coverage Δ Complexity Δ
.../io/fabric8/maven/docker/service/BuildService.java 34.78% <0%> (+3.53%) 9 <0> (ø) ⬇️
...config/handler/property/PropertyConfigHandler.java 90.54% <100%> (ø) 42 <1> (ø) ⬇️
...ic8/maven/docker/access/ContainerCreateConfig.java 77.77% <100%> (+1.15%) 26 <1> (+1) ⬆️
...ain/java/io/fabric8/maven/docker/util/EnvUtil.java 68.83% <75%> (-0.77%) 52 <3> (+6)
.../io/fabric8/maven/docker/config/NetworkConfig.java 91.3% <0%> (-0.5%) 54% <0%> (+27%)
...o/fabric8/maven/docker/log/DefaultLogCallback.java 86.56% <0%> (+4.74%) 15% <0%> (+4%) ⬆️

@rhuss
Copy link
Collaborator

rhuss commented Feb 7, 2018

@stromnet thanks for the PR ! I see clearly your use case and think that is useful.

However, I would strive for another approach: We already have the possibility to make external configuration via properties by adding to an image configuration:

<image>
  <external>
     <type>properties</type>
     <prefix>docker</prefix> <!-- this is the default -->
  </external>
</image>

This means you can fully configure your application (build and runtime aspects) via system properties. As this functionality overlaps which your PR (having a docker.runEnv is clearly part of an image of an image configuration), I do not want to add this at a global level outside this handler.

Currently, the only problems we have that this not already works out of the box are:

  • The property handler only support docker.env.... properties which are used for both, build and runtime. There should be dedicated runEnv and buildEnv kind of properties.
  • When you configure external properties likes this, you cannot use the XML configuration (its an either-or thing). However, it would be easy to allow merging of configuration, that is if also XML config is provided, then it's used as the default, overwritable by properties.

So, ideally, the configuraton should look like:

<image>  
  <external>  <type>properties</type>  </external>
  <name>myimage</name>
  <build> 
       ...
  </build>
   <run>
      ....
   </run>
</image>

Then you can overwrite or add to the configuration by providing the corresponding properties. This also would fit nicely into our current configuration scheme.

If you feel fancy, it would be awesome if you could add this. (happy to help and give pointers). If not, I would tackle this but cannot promise any ETA (so busy until May right now ;-(

@stromnet
Copy link
Contributor Author

stromnet commented Feb 7, 2018

@rhuss, thanks for response!

We've looked at the externalised solution, but since the pom file needs total rewrite to use only params and no XML, we rather avoid that. With customers ~100 projects it would be quite messy (hm.. must have counted wrong somewhere, can't really be that many.. but close!)

A merging of XML parameters with properties would indeed be an useful solution, especially if it can be made without adding additional <external> configuration and touching any pom at all (besides bumping version)... ;) Actually while playing with properties it took a while to understand why setting docker.env. didn't have any effect, before I realised that docker.buildArg was a special case and all others required this externalised configuration. Having merged parameters would feel very natural.

The only gotcha is that we don't want to set http_proxy in <run><env>.. since that would be embedded in the image, so some kind of buildEnv vs runEnv would be required.

Feel free to give some pointers on what would have to be done, and depending on how much work it is I might be able to get some time to do it.

@rhuss
Copy link
Collaborator

rhuss commented Feb 7, 2018

Cool. I think its not so hard:

@stromnet
Copy link
Contributor Author

stromnet commented Feb 7, 2018

Right, ok will try to take a look at it, probably Friday or next week.
We actually want this to set build/run pullPolicy as well, so multiple reasons to get this fixed.

Any preferences on priority, properties overrides XML I guess, and if external is configured we ignore any XML like is done today?

General maven question, not too sure about all the internals: Should we evaluate System properties too (-D flags), I don't think maven merges those into project properties, but is there some other source of those perhaps?

@rhuss
Copy link
Collaborator

rhuss commented Feb 7, 2018

Actually, I really like to have this merging when <external> is configured, and no property handling if no <external> is given by default (as it is now, except with the additional possibility to merge in XML configuration). And right, properties should then take precedence.

I see your use case that you want to have the property provider enabled by default (to not touch existing configuration), but this would break backwards compatibility, and also, what would you do when multiple images are configured ? (therefor you have the possibility to provide some prefix which is Docker by default).

Maybe a good compromise would be to add a new global configuration option enablePropertyConfiguration or so (which you then also could set via a property) and which then leads to your desired behaviour. This config then could also take the global prefix.

Yeah, sorry for docker.buildArgs being handled differently as the other property based configuration. Actually, when we already had this generic merge, I think this wouldn't have been necessary. Probably I will reconsider this option when the merging is in place.

wdyt ?

btw, I on PTO next week, so don't be surprised if I answer only slowly (well, in the recent busy times my response time was already suboptimal already ... :)

@stromnet
Copy link
Contributor Author

stromnet commented Feb 7, 2018

A property such as docker.enablePropertyConfiguration to enable it seems like a good thing. Not adding much complexity to my use-case, and still maintaining the backwards compatibility. Enabling it by default is indeed an issue if you have multiple images in one project, which would all react to docker.xxx properties, so I see the need.

Will give it a try!

@rhuss
Copy link
Collaborator

rhuss commented Feb 8, 2018

Awesome, thanks !

@stromnet
Copy link
Contributor Author

stromnet commented Feb 12, 2018

On good way today, created a separate #941 for run/build env, and started on looking into merging too.

Basically, as you mentioned, I start with RunImageConfiguration.Builder(base), and then for every property merge them somehow. Question is, what makes most sense?

For simple String property, this could work:

public Builder imagePullPolicy(String imagePullPolicy) {
            config.imagePullPolicy = firstNonNull(imagePullPolicy, config.imagePullPolicy);
            return this;
        }

I.e, if it is explicitly set by property (which comes as param here), we use that, else we use that from base (which is null if not set).
But then, there is no way to override XML and set null from property... Should a blank string null it out perhaps?

Then we have for example tags, which takes a List of strings. Should the properties override all tags, or extend the list? For tags both scenarios makes sense.
But for runCmds, you only want either base or properties, not something merged.

Same questions applies to Maps, such as buildOptions, args, env.

Thoughts?

@stromnet stromnet closed this Feb 12, 2018
@stromnet stromnet reopened this Feb 12, 2018
@stromnet
Copy link
Contributor Author

Ok, got some more done now. Here is WIP work, no nice descriptive text here yet but if you have feedback please give it :)

stromnet@e5c137a

@stromnet
Copy link
Contributor Author

Opened new PR in #948

@stromnet stromnet closed this Feb 19, 2018
@stromnet stromnet deleted the runenv-properties branch March 19, 2018 07:35
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.

2 participants