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

Enable parsing the OCI linux configuration on non-linux platforms #135

Closed
wants to merge 3 commits into from

Conversation

laijs
Copy link
Contributor

@laijs laijs commented Sep 2, 2015

The first patch changes the user of the Process to a json.RawMessage

The type of user becomes json.RawMessage instead of platform aware
structure, so the Spec becomes a platform-independent structure.

It makes we can parse the config.json as a Spec before knowing
the os, and allows we move the linux related *.go to linux/.

The runtime should parse the user with the corresponding platform
specific User structure.

The second patch enables parsing the OCI linux configuration on non-linux platforms

What does this patch do?

  • Make the LinuxSpec be visible by golang on non-linux platforms when “go build”

What did this patch change?

  • Add linux/
  • Rename config_linux.go to linux/config.go
  • Rename config-linux.md to linux/config.md
  • Rename runtime_config_linux.go to linux/runtime-config.go
  • Rename runtime-config-linux.md to linux/runtime-config.md
  • Rename runtime-linux.md -> linux/runtime.md
  • Remove the “// +build linux” from linux/config.go
  • The package name of linux/*.go is changed to “linux”
  • Add import of the “specs” in the linux/*.go

The type LinuxSpec (or say the OCI linux configuration) is a structure
describing some information, there is no executable code bound to the linux,
so the LinuxSpec should be naturally usable by any platform when needed.

But the file names (ended with “_linux.go”) and the build tag disallow
us to do so. The patch changes it. Removing the “_linux” and the build tag
makes go build accept the LinuxSpec on any platform. The linux related files
are also moved to the linux/ to eliminate the future possible name clashing
(include the file names and the type names).

After making the LinuxSpec be visible on non-linux platforms, anyone can
do anything of the following in any platforms. (They are also the useful usecases)

    1. parse the OCI linux configuration (config.json & runtime.json)
    1. verify the OCI linux configuration or the bundle
    1. check or test the OCI linux configuration or even the bundle
    1. discover/build/deliver linux OCI bundle
    1. Run the container via the hypervisor-based runtime on non-linux platforms
  • etc.)

The 5) is one thing of what we are focusing on. After this patch applied,
anyone can build the runv[1] and run linux container on darwin. (Also after
we update the opencontainers/spec under Godeps/ in our runv.git[1] project.)

[1]: /~https://github.com/hyperhq/runv

@totherme
Copy link

totherme commented Sep 2, 2015

We would also like to be able to build on darwin, so we're interested in this PR.

What happens if someone tries to add a windows-compatible User type? Will it clash with the User definition in config-linux.go? Should we consider renaming User to LinuxUser to avoid such clashes?

@totherme and @glyn

@laijs
Copy link
Contributor Author

laijs commented Sep 2, 2015

@totherme good question!

I don't want to add LinuxUser. We should move the files to the linux/ directory
to avoid the naming clashes and the window files to windows/.

We can also move them now. @crosbymichael @philips

@glyn
Copy link

glyn commented Sep 2, 2015

@laijs sounds good.

@laijs
Copy link
Contributor Author

laijs commented Sep 2, 2015

The User in the Spec need to be changed to a reference as user string.
and the User in the config-linux.go needs a name for referencing.

It is important for making the Spec be a platform-independent structure.

@laijs laijs force-pushed the linuxspec-on-non-linux branch from 122237a to cd8c885 Compare September 2, 2015 15:03
glyn added a commit to cf-guardian/specs that referenced this pull request Sep 2, 2015
it is now possible to create Spec structs on darwin machines. This may
be superseded by opencontainers#135

This is primarily used by /~https://github.com/cloudfoundry-incubator/nocs
[#101501242] at https://www.pivotaltracker.com/n/projects/1158420

Signed-off-by: Gareth Smith <gsmith@pivotal.io>
glyn added a commit to cloudfoundry-attic/nocs that referenced this pull request Sep 2, 2015
We currently depend on /~https://github.com/cf-guardian/specs which is a
fork of /~https://github.com/opencontainers/specs and which may be
superseded by the pull request
opencontainers/runtime-spec#135.

[#101501242]

Signed-off-by: Gareth Smith <gsmith@pivotal.io>
@mrunalp
Copy link
Contributor

mrunalp commented Sep 4, 2015

Needs rebase.

@philips
Copy link
Contributor

philips commented Sep 4, 2015

I would rather just leave the user field out completely instead of creating a mapping string. Thoughts @crosbymichael . Overall I am fine with this change.

@dqminh
Copy link
Contributor

dqminh commented Sep 4, 2015

I'm not sure about this. It seems that it's more reasonable to add additional data structure for different plarforms instead of having linux structure available to linux platform. Hypervisor / darwin / windows should be able to add their own data structure here, or we can add additional build tags to linux-specific file if the platform supports linux structure.

@laijs
Copy link
Contributor Author

laijs commented Sep 4, 2015

@mrunalp
Filename-renames generally causes conflict quickly, so reviewing the attempts/proposal of the PR (or include the non-rename-part) at first is what I beg for. Anyway I will update the patch soon.

@laijs
Copy link
Contributor Author

laijs commented Sep 4, 2015

@philips The user field is important for the Process. Leaving the user field out may also lead to leave the main-process out from the Spec structure as consequence which is accepted by me. I hope more maintainers give opinions about leaving the main-process out from the Spec structure or other solutions.

@laijs
Copy link
Contributor Author

laijs commented Sep 4, 2015

@dqminh IMO, the more reasonable choice would be the LinuxSpec, WindowsSpec and DarwinSpec can all live together in any platforms. We can't achieve it by build-tags, only well structuring of the code helps.

I don't quit got the meaning of "instead of having linux structure available to linux platform", this pr makes the linux structure available for any platforms including linux.

specs/
         config.go
         linux/
         windows/
import (
    "opencontainers/specs"
    "opencontainers/specs/linux"
    "opencontainers/specs/windows"
)

func useSpec() {
    // parse the config as specs.Spec
   if result.Platform.OS == "linux" {
       // re-parse the config as linux.Spec and use it
       // after my pr accepted, the name is still linux.LinuxSpec, but it it just another orthogonal problem
   } else if result.Platform.OS == "windows" {
       // re-parse the config as windows.Spec and use it
   }
}

@julz
Copy link
Contributor

julz commented Sep 4, 2015

One possibility is to borrow from the Golang approach (in exec.Cmd) and use a common shared struct with a system specific field for system-specific info (SysProcAttr, in the case of exec.Cmd).

This would make Spec the root (instead of Spec being nested in LinuxSpec as today). We would then have a nested PlatformAttr field (for example, there may be better names for this) which would be selected based on the platform. This could be stored as a json.RawMessage and parsed based on the selected platform (or, to state it in terms of json rather than go, the schema of this field would be defined by the selected platform). Something like this:

type Spec struct {
 // ...
 Platform string
 PlatformAttr json.RawMessage
}

type LinuxSpec struct {
   // ...
}

Parsing would then look like this rather than needing to reparse the same file:

 var spec specs.Spec
 json.NewDecoder(file).Decode(&spec)

 select spec.Platform {
   case "Linux":
     var linuxSpec linux.Spec
     json.NewDecoder(spec.PlatformAttr).Decode(&linuxSpec)
  default:
    panic("unsupported platform")
 }

If we wanted to support the case of a container which supports multiple platforms (runC, runV for example), PlatformSpec could be a map[string]json.RawMessage (this avoids having to include the same common spec in multiple files in the case of a multi-platform container).

For the user field, we could have a similar platform-specific json.RawMessage PlatformAttr attribute on the ProcessSpec (whose schema would be based on the relevant Platform), as above.

@laijs
Copy link
Contributor Author

laijs commented Sep 4, 2015

@julz The Spec being the root or being nested in PlatformSpec are the same case when we face the attempt that we want to allow all the PlatformSpecs visible at the same time at any platform. In Both cases we need to move the LinuxSpec to linux/ or add "Linux" prefix to all possible-name-clashing fields.

Using json.RawMessage for the user field is a good suggestion. It avoids the indirectly mapping the platform-specific User. Thanks.

@julz
Copy link
Contributor

julz commented Sep 4, 2015

@laijs indeed, to be clear I was suggesting Spec being the root to avoid double-parsing the config file and to provide a simple potential path to supporting multiple platforms in a bundle (and because to my eyes it's clearer than platform-specific configs with nested common specs and mirrors the golang convention for exec); this is probably an orthogonal issue which may make more sense as another PR though. I'm totally in favour of moving the platform-specific parts to their own packages so that they're usable on any platform.

@laijs
Copy link
Contributor Author

laijs commented Sep 4, 2015

My code in previous comments is just an example, it doesn't mean double-parsing is required in most cases.

In many cases, the platform is known (runc build on linux currently "knows" the config.json is for linux platform), so one parsing is enough. Spec being the root will leads to two-steps parsing.

Yes, it is orthogonal issue.

@laijs laijs force-pushed the linuxspec-on-non-linux branch from cd8c885 to b436080 Compare September 4, 2015 16:08
@laijs
Copy link
Contributor Author

laijs commented Sep 4, 2015

Rebased.
The User is also changed to json.RawMessage.

@laijs
Copy link
Contributor Author

laijs commented Sep 7, 2015

@@ -21,8 +25,8 @@ type Spec struct {
type Process struct {
// Terminal creates an interactive terminal for the container.
Terminal bool `json:"terminal"`
// User specifies user information for the process.
User User `json:"user"`
// User specifies user information for the process. It is platform-specific structure.
Copy link
Contributor

Choose a reason for hiding this comment

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

It is a platform-specific structure.

@mrunalp
Copy link
Contributor

mrunalp commented Sep 8, 2015

Added a minor-nit, otherwise looks good. (Needs another rebase.)

@laijs laijs force-pushed the linuxspec-on-non-linux branch from b436080 to 7dd3a6c Compare September 9, 2015 00:57
@laijs
Copy link
Contributor Author

laijs commented Sep 9, 2015

@mrunalp updated as you suggested. Thx!

Mrunal Patel and others added 3 commits September 9, 2015 08:17
The type of `user` becomes `json.RawMessage` instead of platform aware
structure, so the Spec becomes a platform-independent structure.

It makes we can parse the config.json as a Spec before knowing
the os, and allows we move the linux related *.go to linux/.

The runtime should parse the user with the corresponding platform
specific User structure.

Signed-off-by: Lai Jiangshan <jiangshanlai@gmail.com>
What does this patch do?
 * Make the LinuxSpec be visible by golang on non-linux platforms when “go build”

What did this patch change?
 * Add linux/
 * Rename config_linux.go to linux/config.go
 * Rename config-linux.md to linux/config.md
 * Rename runtime_config_linux.go to linux/runtime-config.go
 * Rename runtime-config-linux.md to linux/runtime-config.md
 * Rename runtime-linux.md -> linux/runtime.md
 * Remove the “// +build linux” from linux/config.go
 * The package name of linux/*.go is changed to “linux”
 * Add import of the “specs” in the linux/*.go

The type LinuxSpec (or say the OCI linux configuration) is a structure
describing some information, there is no executable code bound to the linux,
so the LinuxSpec should be naturally usable by any platform when needed.

But the file names (ended with “_linux.go”) and the build tag disallow
us to do so. The patch changes it. Removing the “_linux” and the build tag
makes `go build` accept the LinuxSpec on any platform. The linux related files
are also moved to the linux/ to eliminate the future possible name clashing
(include the file names and the type names).

After making the LinuxSpec be visible on non-linux platforms, anyone can
do anything of the following in any platforms. (They are also the useful usecases)
 * 1) parse the OCI linux configuration (config.json & runtime.json)
 * 2) verify the OCI linux configuration or the bundle
 * 3) check or test the OCI linux configuration or even the bundle
 * 4) discover/build/deliver linux OCI bundle
 * 5) Run the container via the hypervisor-based runtime on non-linux platforms
 * etc.)

The 5) is one thing of what we are focusing on. After this patch applied,
anyone can build the runv[1] and run linux container on darwin. (Also after
we update the opencontainers/spec under Godeps/ in our runv.git[1] project.)

[1]: /~https://github.com/hyperhq/runv

Signed-off-by: Lai Jiangshan <jiangshanlai@gmail.com>
@laijs laijs force-pushed the linuxspec-on-non-linux branch from 7dd3a6c to e3028f5 Compare September 9, 2015 16:22
@laijs
Copy link
Contributor Author

laijs commented Sep 9, 2015

Rebased again, on the top of newest master (the latest merge of #162). Thx!

@vbatts
Copy link
Member

vbatts commented Sep 9, 2015

@laijs I think we're going to take this conversation a little more broad than getting into this approach. see #166

@laijs
Copy link
Contributor Author

laijs commented Sep 16, 2015

This pr will be pending (not close) until the user is changed to platform-agnostic.
I will update this pr properly after the "user" is fixed.
I hope and assume the "user" problem be fixed before v0.2.0

@laijs
Copy link
Contributor Author

laijs commented Sep 16, 2015

#191

glyn added a commit to cf-guardian/specs that referenced this pull request Sep 22, 2015
it is now possible to create Spec structs on darwin machines. This may
be superseded by opencontainers#135

This is primarily used by /~https://github.com/cloudfoundry-incubator/nocs
[#101501242] at https://www.pivotaltracker.com/n/projects/1158420

Signed-off-by: Gareth Smith <gsmith@pivotal.io>
// User specifies user information for the process.
User User `json:"user"`
// User specifies user information for the process. It is a platform-specific structure.
User json.RawMessage `json:"user"`
Copy link
Contributor

Choose a reason for hiding this comment

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

Can User be a string?

Copy link
Contributor

Choose a reason for hiding this comment

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

On Wed, Dec 16, 2015 at 10:25:08AM -0800, Vish Kannan wrote:

@@ -21,8 +25,8 @@ type Spec struct {
type Process struct {
// Terminal creates an interactive terminal for the container.
Terminal bool json:"terminal"

  • // User specifies user information for the process.
  • User User json:"user"
  • // User specifies user information for the process. It is a platform-specific structure.
  • User json.RawMessage json:"user"

Can User be a string?

User as a string requires /etc/passwd lookups or some such on Linux,
and I don't think we want to require that (although I'm fine with it
as an option). See my suggestions in #191 for an alternative that
allows both name-based and number-based identification 1, but that's
going to be platform-specific.

@mrunalp mrunalp added this to the v0.3.0 milestone Jan 5, 2016
@mrunalp
Copy link
Contributor

mrunalp commented Jan 13, 2016

We will handle this as part of unifying the runtime/config json files.

@mrunalp mrunalp closed this Jan 13, 2016
@wking
Copy link
Contributor

wking commented Jan 13, 2016

On Wed, Jan 13, 2016 at 02:05:21PM -0800, Mrunal Patel wrote:

We will handle this as part of unifying the runtime/config json
files.

Those seem orthogonal to me (e.g. #284 unifies configs but does not
address cross-platform Go parsing). And #284 is enough of a bear to
rebase and review that I'd rather not have it biting off more than it
needs to.

vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Jan 13, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

Not adding a "name" string, as that is not a requirement yet.

In the event a windows runtime shows up, I could imagine an `sid`, but
we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Jan 14, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Jan 14, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Jan 15, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Jan 15, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Jan 15, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Mar 9, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
vbatts added a commit to vbatts/oci-runtime-spec that referenced this pull request Mar 9, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
Mashimiao pushed a commit to Mashimiao/specs that referenced this pull request Aug 19, 2016
This introduces verbiage of fields that may occur in json (technically
optional), but is required on certain platforms (e.g. Linux).

The JSON document will look the same as it presently does, but now the
reference source compiles regardless of platform.

Not adding a "name" string to the user sturct, as that is not a
requirement yet.

In the event a windows runtime shows up, I could imagine an `sid` on the
user struct, but we'll get to that when it happens.

Closes opencontainers#135
Related to opencontainers#166

Signed-off-by: Vincent Batts <vbatts@hashbangbash.com>
@wking wking mentioned this pull request Sep 26, 2016
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.

10 participants