-
Notifications
You must be signed in to change notification settings - Fork 204
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
containers.conf: appendable string arrays, Part 1
Commit 6506f4f implemented a POC to allow for changing the behavior when loading multiple container.conf files in sequence. By default, the TOML encoder will override existing data/fields with the one specified in the loaded file. The POC has demonstrated how this behavior can be changed to append string slices instead of overriding/replacing them entirely. This change is the first step of integrating these appendable string arrays into containers.conf and starts with enabling the `Env`, `Mounts`, and `Volumes` fields in the `[Containers]` table. Signed-off-by: Valentin Rothberg <[email protected]>
- Loading branch information
Showing
11 changed files
with
147 additions
and
113 deletions.
There are no files selected for viewing
73 changes: 0 additions & 73 deletions
73
internal/attributed-string-slice/attributed_string_slice.go
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
package attributedstring | ||
|
||
import ( | ||
"bytes" | ||
"fmt" | ||
|
||
"github.com/BurntSushi/toml" | ||
) | ||
|
||
// Slice allows for extending a TOML string array with custom | ||
// attributes that control how the array is marshaled into a Go string. | ||
// | ||
// Specifically, an Slice can be configured to avoid it being | ||
// overridden by a subsequent unmarshal sequence. When the `append` attribute | ||
// is specified, the array will be appended instead (e.g., `array=["9", | ||
// {append=true}]`). | ||
type Slice struct { // A "mixed-type array" in TOML. | ||
// Note that the fields below _must_ be exported. Otherwise the TOML | ||
// encoder would fail during type reflection. | ||
Values []string | ||
Attributes struct { // Using a struct allows for adding more attributes in the future. | ||
Append *bool // Nil if not set by the user | ||
} | ||
} | ||
|
||
// Get returns the Slice values or an empty string slice. | ||
func (a *Slice) Get() []string { | ||
if a.Values == nil { | ||
return []string{} | ||
} | ||
return a.Values | ||
} | ||
|
||
// UnmarshalTOML is the custom unmarshal method for Slice. | ||
func (a *Slice) UnmarshalTOML(data interface{}) error { | ||
iFaceSlice, ok := data.([]interface{}) | ||
if !ok { | ||
return fmt.Errorf("unable to cast to interface array: %v", data) | ||
} | ||
|
||
var loadedStrings []string | ||
for _, x := range iFaceSlice { // Iterate over each item in the slice. | ||
switch val := x.(type) { | ||
case string: // Strings are directly appended to the slice. | ||
loadedStrings = append(loadedStrings, val) | ||
case map[string]interface{}: // The attribute struct is represented as a map. | ||
for k, v := range val { // Iterate over all _supported_ keys. | ||
switch k { | ||
case "append": | ||
boolVal, ok := v.(bool) | ||
if !ok { | ||
return fmt.Errorf("unable to cast append to bool: %v", k) | ||
} | ||
a.Attributes.Append = &boolVal | ||
default: // Unsupported map key. | ||
return fmt.Errorf("unsupported key %q in map: %v", k, val) | ||
} | ||
} | ||
default: // Unsupported item. | ||
return fmt.Errorf("unsupported item in attributed string slice: %v", x) | ||
} | ||
} | ||
|
||
if a.Attributes.Append != nil && *a.Attributes.Append { // If _explicitly_ configured, append the loaded slice. | ||
a.Values = append(a.Values, loadedStrings...) | ||
} else { // Default: override the existing Slice. | ||
a.Values = loadedStrings | ||
} | ||
return nil | ||
} | ||
|
||
// MarshalTOML is the custom marshal method for Slice. | ||
func (a *Slice) MarshalTOML() ([]byte, error) { | ||
iFaceSlice := make([]interface{}, 0, len(a.Values)) | ||
|
||
for _, x := range a.Values { | ||
iFaceSlice = append(iFaceSlice, x) | ||
} | ||
|
||
if a.Attributes.Append != nil { | ||
Attributes := make(map[string]any) | ||
Attributes["append"] = *a.Attributes.Append | ||
iFaceSlice = append(iFaceSlice, Attributes) | ||
} | ||
|
||
buf := new(bytes.Buffer) | ||
enc := toml.NewEncoder(buf) | ||
if err := enc.Encode(iFaceSlice); err != nil { | ||
return nil, err | ||
} | ||
return buf.Bytes(), nil | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.