-
Notifications
You must be signed in to change notification settings - Fork 208
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
Advanced docker runtime #2167
base: release/v1
Are you sure you want to change the base?
Advanced docker runtime #2167
Conversation
Signed-off-by: Rich Baird <[email protected]>
Signed-off-by: Rich Baird <[email protected]>
Signed-off-by: Rich Baird <[email protected]>
Signed-off-by: Rich Baird <[email protected]>
…ion options Signed-off-by: Rich Baird <[email protected]>
Signed-off-by: Rich Baird <[email protected]>
4fe1db6
to
9c1f3e1
Compare
It looks like the bundle defines both the mount point into the container AND the source from the host. Am I misunderstanding or how does the user specify what they want to expose to the bundle? |
That is correct. This has the implication that the source has to be the same on both the build and deployment environments. In my particular case I think that's probably okay, but it shouldn't be too difficult to add a CLI option to overwrite the path on the host if that's a more desirable implementation. |
It's a hard requirement in Porter that bundles do not have knowledge of the host machine. So Porter needs a way to define mappings from what the bundle has declared (volumes, mounts, etc) and how those should be configured by the host when running the container. In general, we can't optimize for "in-house" bundles. Any new feature needs to work when the bundle author and the people running the bundle are different entities. For example, a software vendor has a bundle that installs their product and their users should be able to run it without hacks or mimicing the vendor's environment. |
I recommend working out and getting consensus on how this feature should work before putting more time into a PR. Let's figure out how we want the bundle author to define additional configuration, if it's required, etc and then how the person running the bundle would be able to learn what configuration is required, what their options are (i.e. hey this bundle exposes a mount point), and how they would configure it when running the bundle. |
cf4a6c8
to
3bd1596
Compare
… advanced-docker-runtime
That sounds great. I have put some effort into modifying the implementation to accept parameters. It's not quite working. I can tell that the template strings are being matched because they're being replaced with an empty string, but I am obviously missing something about how the parameter values are retrieved. You can take a look at the changes on my repo: richbai90#1 . If I can get a little more direction, I think I can make this work, if it seems like a positive direction to you. |
Sorry, I am having trouble telling what you are trying to do in that PR. One of the reasons why this wasn't implemented previously is because it requires an agreed upon design for how a bundle author specifies mounts, ideally in a way that works with our existing patterns (parameters) and can be promoted to the CNAB spec. It shouldn't just be "we are exposing docker functionality without an abstraction layer, since long term porter will support more than k8s and docker, for example virtual machines or web assembly modules. Parameters let the bundle define what it needs, and then gives the end-user of the bundle flexibility to choose how to supply that parameter, such as providing it from a secret vs an environment variable. Porter defines special parameter types that simplify passing data into bundles as parameters. Right now we support string, bool, number, file, etc. What if we added a new parameter type: directory and then the user could choose to satisfy it with a volume mount? # porter.yaml
parameters:
- name: nginx-config
type: directory
path: /etc/nginx
mode: read # or read-write if we are writing back to the host Then when the end-user of the bundle checks out the bundle to see how they can customize how it's installed they would see something like this: $ porter explain -r MYBUNDLE
Name: mybundle
Description: An example bundle that mounts a directory
Version: 0.1.0
Porter Version: v1.0.0-alpha.19
Parameters:
-----------------------------------------------------------------
Name Description Type Default Required Applies To
-----------------------------------------------------------------
nginx-config directory true All Actions Then the user has their choice of how they want to provide that parameter. # pass in the contents of /tmp/myconfig from the host
porter install --param nginx-config=/tmp/myconfig name: mybundle-params
parameters:
- name: nginx-config
source:
# could specify whatever the docker --mount flag supports
# target isn't required since Porter can determine that info from the parameter definition
mount: source=/home/myuser/myconfig Another design considering is how we would deal with file permissions and the owner/group on any files created. Porter runs as a random nonroot user, and we would need to think through how to avoid predictable problems caused by the container writing files to the host as that user. This isn't everything that needs to be taken into account in a design but I hope that gives you an idea of what we need before the proposal can be a accepted, and then an implementation considered. |
The goal was to be able to include parameters in the extension definition the same way they are used in the mixin definitions. For example: required:
docker:
mounts:
- source: {{ mySource }} I'm good with however you feel the best way to implement this is. My thought was eventually to add a new runtime:
docker:
mounts:
- source: {{ mySource }} As for the owner/group considerations, you can see how I'm using the setup currently and handling these considerations with my file-xfer mixin. Basically, the mixin works in four steps:
The archive method avoids the problems of ownership changes of the individual files because only the archive owner changes during the build process. During the install process the available buildargs can be used to appropriately set the |
This makes a lot of sense to me. I can work on a PR to that affect. |
required:
docker:
mounts:
- source: {{ mySource }} With the solution I proposed above, using parameters to mount volumes, this part shouldn't be needed anymore. Do you agree? |
I'm not too concerned about how the xfer mixin deals with file permissions. Rather it's the general case of a bundle mounting a volume and then writing files to it that needs further planning. Porter will need to provide strong guidance on how to interact with a volume, backed with helpers that make it easy to just do the right thing without reimplementing it in each mixin/bundle. For example, if I have a bundle that mounts a volume from the host, and then I write to that volume with the exec mixin (like in a bash script), what is the desired experience for the bundle author so that it's super easy for them to get the permissions right? We want to avoid making it easy for bundle authors to mess up someone's file permissions because we've made it 100% their responsibility. I think we have a bunch of options for how to go about this (off the top of my head):
I strongly lean towards the first option if possible, since Porter is somewhat responsible for not giving bundle authors a "foot gun" to mess up permissions on people's machines. 😀 |
All of this sounds very reasonable. To make sure I understand, the current consideration is something like this.
Does that sound more or less like what we're looking for in terms of features? Regarding this feature:
I waffled some on whether to include it, or if the end user should always have control over the files written to their machine. I determined to include it for instances where the author or developer knows best. For instance, if the bundle is expected to interact with the alsa kernel module, there is a good chance that any files the bundle writes to the host will need to be owned by the Does it make sense to only allow bundle authors to specify system users/groups as file owners? IE put a validation step in the porter install command that checks if the user/group exists and is a system user/group. |
I do agree. I think this is the best abstraction. It should work regardless of runtime used, and doesn't expose unnecessary, potentially dangerous docker configuration options. It also abstracts the details of implementation away from bundle authors who shouldn't need to be concerned with them. |
Agreed, as long as the bundle does not mount the docker socket (using requires docker). If the end user runs a bundle with --allow-docker-host-access, honestly that is a security concern that we call out on that flag and is why the end user has to explicitly allow it. But if they are only using the directory parameter, then yes their access to the rest of the host is limited to just the volumes mounted.
I really lean towards having a default uid/gid defined on the parameter, and then porter has a way for the end user to always be able to override that if necessary. That way the bundle author can suggest a default, but if there's a mismatch, the end user should ultimately always have control over what is written to the host volume. # porter.yaml
parameters:
- name: nginx-config
type: directory
path: /etc/nginx
writable: true # Replaces my earlier suggestion of read|read-write, defaults to false
gid: 1001 # The bundle author can default it to a well-known user/group
uid: 1001 So if the user has a custom setup, and needs a different user, they could run
The bundle would have access to these additional gid/uid parameters if the bundle wants to set owners (not necessary but for situations where the underlying tools requires that info), but Porter would always do a sweep at the end to enforce ownership on all shared volumes.
I'm not sure I see how whether or not the user is flagged as a system user is helpful here and it most likely assumes too much about how people organize their users/groups, and assumes Linux (we don't support Windows containers now but could in the future). |
Thanks for the use cases listed out like that! 👍 |
Yeah no sweat! That's how we write stories where I work. I'll get to work on a PR to this affect. |
I have a mostly working implementation of what we discussed on #2251. I am not sure where to look for the parameter default overwriting via the command line logic. That is what is left not working currently. |
What does this change
Add additional options to the docker extension within porter to permit the configuration of mounts, network types, and kernel capabilities. The new configuration follows the API established in the
go-cnab
package. The one downside to this is that the new configuration options use the proper case format go uses to identify package exports. This seems unavoidable until thego-cnab
package updates the driver configuration on their side to include appropriate tags to identify the fields in their JSON naming convention.The change adds additional options to the docker extension. The complete configuration is featured below, however some options cannot be specified with others according to the docker driver spec as implemented by the
go-cnab
package.What issue does it fix
This is in reference to #2125. It doesn't address the proposal as such, but is a proposal of how the docker runtime environment could be expanded as per the comment #2125 (comment)
Notes for the reviewer
In this case no changes had to be made to to the docker driver defined in the
go-cnab
package since that driver exposes all the docker api options provided by the official docker package. Porter just had to be extended to expose additional options through its configuration options.Checklist
Reviewer Checklist