Documentation ¶
Overview ¶
Package genutil provides utility functions for package that generate code based on a YANG schema.
Index ¶
- Constants
- func CallerName() string
- func CamelCaseNameExt(exts []*yang.Statement) (string, bool)
- func EntryCamelCaseName(e *yang.Entry) string
- func FindAllChildren(e *yang.Entry, compBehaviour CompressBehaviour) (map[string]*yang.Entry, []error)
- func GetOrderedEntryKeys(entries map[string]*yang.Entry) []string
- func MakeNameUnique(name string, definedNames map[string]bool) string
- func OpenFile(fn string) *os.File
- func ParentModuleName(node yang.Node) string
- func ParentModulePrettyName(node yang.Node) string
- func SyncFile(fh *os.File)
- func TypeDefaultValue(t *yang.YangType) *string
- func WriteIfNotEmpty(b io.StringWriter, s string)
- type CompressBehaviour
Constants ¶
const ( // GoDefaultYgotImportPath is the default import path used for the ygot library // in the generated code. GoDefaultYgotImportPath string = "github.com/openconfig/ygot/ygot" // GoDefaultYtypesImportPath is the default import path used for the ytypes library // in the generated code. GoDefaultYtypesImportPath string = "github.com/openconfig/ygot/ytypes" // GoDefaultGoyangImportPath is the default path for the goyang/pkg/yang library that // is used in the generated code. GoDefaultGoyangImportPath string = "github.com/openconfig/goyang/pkg/yang" // GoDefaultGNMIImportPath is the default import path that is used for the gNMI generated // Go protobuf code in the generated output. GoDefaultGNMIImportPath string = "github.com/openconfig/gnmi/proto/gnmi" )
Variables ¶
This section is empty.
Functions ¶
func CallerName ¶
func CallerName() string
CallerName returns the name of the Go binary that is currently running.
func CamelCaseNameExt ¶
CamelCaseNameExt returns the CamelCase name from the slice of extensions, if one of the extensions is named "camelcase-name". It returns the a string containing the name if the bool return argument is set to true; otherwise no such extension was specified. TODO(wenbli): add unit test
func EntryCamelCaseName ¶
EntryCamelCaseName returns the camel case version of the Entry Name field, or the CamelCase name that is specified by a "camelcase-name" extension on the field. The returned name is not guaranteed to be unique within any context.
func FindAllChildren ¶
func FindAllChildren(e *yang.Entry, compBehaviour CompressBehaviour) (map[string]*yang.Entry, []error)
FindAllChildren finds the data tree elements that are children of a YANG entry e, which should have code generated for them. In general, this means data tree elements that are directly connected to a particular data tree element; however, when compression of the schema is enabled then recursion is required.
For example, if we have a YANG tree:
/interface (list) /interface/config (container) /interface/config/admin-state (leaf) /interface/state (container) /interface/state/admin-state (leaf) /interface/state/oper-state (leaf) /interface/state/counters (container) /interface/state/counters/in-pkts (leaf) /interface/ethernet/config (container) /interface/ethernet/config/mac-address (leaf) /interface/ethernet/state (state) /interface/ethernet/state/mac-address (leaf) /interface/subinterfaces (container) /interface/subinterfaces/subinterface (list)
With compression disabled, then each directly connected child of a container should have code generated for it - so therefore we end up with:
/interface: config, state, ethernet, subinterfaces /interface/config: admin-state /interface/state: admin-state, oper-state, counters /interface/state/counters: in-pkts /interface/ethernet: config, state /interface/ethernet/config: mac-address /interface/ethernet/state: mac-address /interface/subinterfaces: subinterface
This is simply achieved by examining the directory provided by goyang (e.Dir) and extracting the direct children that exist. These are appended to the directChildren map (keyed on element name) and then returned.
When compression is on, then more complex logic is required based on the OpenConfig path rules. In this case, the following "look-aheads" are implemented:
The 'config' and 'state' containers under a directory are removed. This is because OpenConfig duplicates nodes under config and state to represent intended versus applied configuration. In the compressed schema then we need to drop one of these configuration leaves (those leaves that are defined as the set under the 'state' container that also exist within the 'config' container), and compressBehaviour specifies which one to drop. The logic implemented is to recurse into the config container, and select these leaves as direct children of the original parent. Any leaves that do not exist in the 'config' container but do within 'state' are operation state leaves, and hence are also mapped.
Above, this means that /interfaces/interface has the admin-state and oper-state as direct children.
Since containers can exist under the 'state' container, then these containers are also considered as direct children of e.
Surrounding containers for lists are removed - that is to say, in an OpenConfig schema a list (e.g. /interface/subinterfaces/subinterface) always has a container that surrounds it. This is due to implementation requirements when it is supported on vendor devices. However, to a developer this looks like stuttering, and hence we remove this - by checking that for each directory that would be a child of e, if it has only one child, which is a list, then we skip over it.
Implementing these two rules means that the schema is simplified, such that the tree described becomes:
/interface: admin-state, oper-state, counters, ethernet, subinterface /interface/counters: in-pkts /interface/ethernet: mac-address
As can be seen the advantage of this compression is that the set of entities for which code generation is done is smaller, with less levels of schema hierarchy. However, it depends upon a number of rules of the OpenConfig schema. If compression is on but the schema does not comply with the rules of OpenConfig schema, then errors may occur and be returned in the []error slice by findAllChildren.
It should be noted that special handling is required for choice and case - because these are directories within the resulting schema, but they are not data tree nodes. So for example, we can have:
/container/choice/case-one/leaf-a /container/choice/case-two/leaf-b
In this tree, "choice" and "case-one" (which are choice and case nodes respectively) are not valid data tree elements, so we recurse down both of the branches of "choice" to return leaf-a and leaf-b. Since choices can be nested (/choice-a/choice-b/choice-c/case-a), and can have multiple data nodes per case, then the addNonChoiceChildren function will find the first children of the specified node that are not choice or case statements themselves (i.e., leaf-a and leaf-b in the above example).
The .*ExcludeDerivedState compress behaviour options further filters the returned set of children based on their YANG 'config' status. When set, then any read-only (config false) node is excluded from the returned set of children. The 'config' status is inherited from a entry's parent if required, as per the rules in RFC6020.
func GetOrderedEntryKeys ¶
GetOrderedEntryKeys returns the keys of a map of *yang.Entry in alphabetical order.
func MakeNameUnique ¶
MakeNameUnique makes the name specified as an argument unique based on the names already defined within a particular context which are specified within the definedNames map. If the name has already been defined, an underscore is appended to the name until it is unique.
func OpenFile ¶
OpenFile opens a file with the supplied name, logging and exiting if it cannot be opened.
func ParentModuleName ¶
ParentModuleName returns the name of the module or submodule that defined the supplied node.
func ParentModulePrettyName ¶
ParentModulePrettyName returns the name of the module that defined the yang.Node supplied as the node argument. If the discovered root node of the node is found to be a submodule, the name of the parent module is returned. If the root has a camel case extension, this is returned rather than the actual module name.
func TypeDefaultValue ¶
TypeDefaultValue returns the default value of the type t if it is specified. nil is returned if no default is specified.
func WriteIfNotEmpty ¶
func WriteIfNotEmpty(b io.StringWriter, s string)
WriteIfNotEmpty writes the string s to b if it has a non-zero length.
Types ¶
type CompressBehaviour ¶
type CompressBehaviour int64
CompressBehaviour specifies how the set of direct children of any entry should determined. Compression indicates whether paths should be compressed in the output of an OpenConfig schema; however, there are different ways of compressing nodes.
const ( // Uncompressed does not compress the generated code. This means list // containers and config/state containers are retained in the generated // code. Uncompressed CompressBehaviour = iota // UncompressedExcludeDerivedState excludes config false subtrees in // the generated code. UncompressedExcludeDerivedState // PreferIntendedConfig generates only the "config" version of a field // when it exists under both "config" and "state" containers of its // parent YANG model. If no conflict exists between these containers, // then the field is always generated. PreferIntendedConfig // PreferOperationalState generates only the "state" version of a field // when it exists under both "config" and "state" containers of its // parent YANG model. If no conflict exists between these containers, // then the field is always generated. PreferOperationalState // ExcludeDerivedState excludes all values that are not writeable // (i.e. config false), including their children, from the generated // code output. ExcludeDerivedState )
Why use an enum? There are 3 dimensions here: compress|preferState|excludeDerivedState No dimension spans across all others' options, so can't extract any out without having to error out for some combinations.
func TranslateToCompressBehaviour ¶
func TranslateToCompressBehaviour(compressPaths, excludeState, preferOperationalState bool) (CompressBehaviour, error)
TranslateToCompressBehaviour translates the set of (compressPaths, excludeState, preferOperationalState) into a CompressBehaviour. Invalid combinations produces an error.
func (CompressBehaviour) CompressEnabled ¶
func (c CompressBehaviour) CompressEnabled() bool
CompressEnabled is a helper to query whether compression is on.
func (CompressBehaviour) StateExcluded ¶
func (c CompressBehaviour) StateExcluded() bool
StateExcluded is a helper to query whether derived state is excluded.