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

[WIP] Add support for Config{} to kubecfg #987

Closed
wants to merge 1 commit into from

Conversation

mfojtik
Copy link
Contributor

@mfojtik mfojtik commented Aug 21, 2014

This PR will allow users to use simple JSON file with arrays of Kubernetes objects (services, pods, replicationControllers) as a parameter for kubecfg -c config.json apply command.

The example JSON file in examples/config/guestbook.json contains all objects required for deploying the guestbook example.

@mfojtik
Copy link
Contributor Author

mfojtik commented Aug 21, 2014

@brendanburns @smarterclayton fyi.

@mfojtik mfojtik changed the title Add support for Config{} to kubecfg [WIP] Add support for Config{} to kubecfg Aug 21, 2014
}

obj := api.Config{}
if err := gojson.Unmarshal(data, &obj); err != nil {
Copy link
Contributor

Choose a reason for hiding this comment

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

Should be yaml parsing to be consistent.

@smarterclayton
Copy link
Contributor

This is to serve as a root for a conversation about how groups of resources in the Kubernetes system can be created, updated, and maintained over time. The simplest possible manifestation is "create a bunch of API objects from a static definition". Higher levels of abstraction (templatizing, generating config, and how config can be transformed in reaction to other changes in the system) ultimately must all be transformed into concrete API definitions.

@smarterclayton
Copy link
Contributor

@bgrant0607 as well

@@ -478,3 +478,9 @@ type WatchEvent struct {
type APIObject struct {
Object interface{}
}

type Config struct {
Services []Service `yaml:"services,omitempty" json:"services,omitempty"`
Copy link
Member

Choose a reason for hiding this comment

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

We have an api.XList object already, for all of these. Might make sense to use it.

@bgrant0607
Copy link
Member

Meta comment: These kinds of changes are hard to reverse engineer later. This MUST be accompanied by documentation, probably in docs/cli.md.

verbs: apply doesn't make sense to me. I'd like existing kubecfg verbs to work on the objects described in the file: create, update, even delete.

selectivity: It doesn't have to be in the initial version, but I'd like to be able to selectively apply the verbs, say to just replicationControllers (so, by type), or to just 1 object, by name.

file structure: 1 array per object type isn't going to fly long-term. It's not composable. I'd like to be able to decompose objects into 1 file per microservice and then just concatenate them together.

api endpoint specification: Related to file structure, but worth calling out. As discussed in #635, we need to think about the API surface as not being monolithic. The APIs may be served by different services eventually, and certainly we'd like to support multiple versions -- a user might copy a redis example using v1 of the API and may write their own php config using v2. Therefore, we need to be able to specify API endpoints in the file, with defaults. Again, this doesn't have to be in the initial version, but we need to keep it in mind.

creation-order dependencies: For the most part, we've tried to avoid the need for a particular creation order, due to loose coupling via label selectors. However, I imagine this may change eventually. Creation in sequential order is possible, but ultimately limiting. Anything else, however, requires object cross references. I'd recommend that we just create all objects in parallel for now.

code structure: Not specific to this PR, but I'd eventually like kubecfg to evolve into a simple command wrapper around a client library that could be used independently of the command, to facilitate programmatic use of the k8s API.

I'm going to be starting on a more comprehensive config proposal today.

createObject(c, pod)
}
for _, replicationController := range obj.ReplicationControllers {
createObject(c, replicationController)
Copy link
Member

Choose a reason for hiding this comment

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

We really need to think about ordering for something like this. Some pods may depend on some services. Can the pod handle it if it starts before its dependencies? Clearly the right answer is "yes it should", but only the human setting up the system knows this.

I suppose it's probably fine to say that this only works for pods where start order doesn't matter. The human can always sort their services/pods into batches and start each batch with this method. But we should explicitly say this somewhere.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is also related to health-checking. How do you know that the service is 'ready'? For example the official mysql Docker image, when started it will first start the mysql to create users/init database and then it will stop and start to run the mysql daemon.

Other option is to assume that containers/images are resilient to service failure and so it will be the container responsibility to poll the services.

Copy link
Member

Choose a reason for hiding this comment

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

I think for now it is probably sufficient to start the services first (as Brian suggests below). Starting the service will cause the appropriate environment variables to get set, even if the service isn't live yet; users of the service would have to retry if it's not up yet.

@smarterclayton
Copy link
Contributor

verbs: apply doesn't make sense to me. I'd like existing kubecfg verbs to work on the objects described in the file: create, update, even delete.

So in your config you'd say "create / update / delete" (objects described in the file)? So a config is a description of operations, rather than declarative? Or are you just saying that you'd create/update/delete a config, with that operation globally applicable, and then have filters on the CLI that would allow targeting specific types (or labels)?

@smarterclayton
Copy link
Contributor

api endpoint specification: Related to file structure, but worth calling out. As discussed in #635, we need to think about the API surface as not being monolithic. The APIs may be served by different services eventually, and certainly we'd like to support multiple versions -- a user might copy a redis example using v1 of the API and may write their own php config using v2. Therefore, we need to be able to specify API endpoints in the file, with defaults. Again, this doesn't have to be in the initial version, but we need to keep it in mind.

We'd actually probably like to be able to do this sooner rather than later so we can contribute objects that aren't in the core api service (a short term object for us).

@lavalamp
Copy link
Member

@bgrant0607:

file structure: 1 array per object type isn't going to fly long-term. It's not composable. I'd like to be able to decompose objects into 1 file per microservice and then just concatenate them together.
... Therefore, we need to be able to specify API endpoints in the file, with defaults.

Yeah, absolutely- the APIObjectList approach I suggested is probably the right direction for this. Version is encoded in the api object directly that way, so there's not a need to specify the api endpoint. (I'm thinking that plugin apis will register themselves such that apiserver acts as a proxy (or redirects) so there is a single, easily discoverable location for each api endpoint.

code structure: Not specific to this PR, but I'd eventually like kubecfg to evolve into a simple command wrapper around a client library that could be used independently of the command, to facilitate programmatic use of the k8s API.

+1; we already have a kubecfg package and a client package. As much of this PR as possible should be pushed into those packages, probably the kubecfg package. The client package is pretty nice apart from the monolithic interface (but is kinda low level), kubecfg package is not quite as nice but we can evolve it to make it more useful.

@smarterclayton
Copy link
Contributor

(I'm thinking that plugin apis will register themselves such that apiserver acts as a proxy (or redirects) so there is a single, easily discoverable location for each api endpoint.

As a simplest step we can assume that the specification of that is outside the config and then it's either a client concern (explicitly list endpoints) or a proxy concern (the global api proxy)

@bgrant0607
Copy link
Member

@smarterclayton

you'd create/update/delete a config, with that operation globally applicable, and then have filters on the CLI that would allow targeting specific types (or labels)?

Yes, this is what I meant.

@bgrant0607
Copy link
Member

@lavalamp Re. ordering: See my comment re. dependencies. Yes, some pods will depend on services. Ideally, if they crashed due to the service environment variables not being set they could restart until they saw valid values. One problem I see with this is how the environment variables are propagated to containers. It might or might not work now, but would not work if we were to take advantage of restarts in Docker. One simple thing we could do is to create all services first, before other types of objects. We could hardcode this behavior initially, but it would be nice for the creation order to be configurable by object type. For example, if there were image builds, we'd want to do all the builds first.

@lavalamp
Copy link
Member

@bgrant0607 Yeah, agree that we need to start all the services first. This may not be sufficient in the future but it's simple and will work for a while.

@thockin
Copy link
Member

thockin commented Aug 24, 2014

I'm 100% in support of the idea, but I am not sure about the details of it. This is one of those things that I think is better discussed in the abstract before quibbling over code.

@smarterclayton
Copy link
Contributor

Discussion then in #1007

@@ -29,6 +29,8 @@ import (
"text/template"
"time"

goyaml "gopkg.in/v1/yaml"
Copy link
Contributor

Choose a reason for hiding this comment

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

no need to name this nothing else is called yaml, afaik.

@lavalamp
Copy link
Member

Closing this for now. Please reopen when discussion in #1007 winds down!!

@lavalamp lavalamp closed this Aug 26, 2014
@mfojtik mfojtik deleted the config branch October 22, 2014 12:19
vishh pushed a commit to vishh/kubernetes that referenced this pull request Apr 6, 2016
b3atlesfan pushed a commit to b3atlesfan/kubernetes that referenced this pull request Feb 5, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants