/test/e2e/kubernetes/node/node.go

https://bitbucket.org/afawkes/acs-engine · Go · 179 lines · 148 code · 17 blank · 14 comment · 34 complexity · fde88a5c59c5566d9deae5f64bc9f596 MD5 · raw file

  1. package node
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "log"
  7. "os/exec"
  8. "regexp"
  9. "strings"
  10. "time"
  11. "github.com/Azure/acs-engine/test/e2e/kubernetes/util"
  12. )
  13. const (
  14. //ServerVersion is used to parse out the version of the API running
  15. ServerVersion = `(Server Version:\s)+(v\d+.\d+.\d+)+`
  16. )
  17. // Node represents the kubernetes Node Resource
  18. type Node struct {
  19. Status Status `json:"status"`
  20. Metadata Metadata `json:"metadata"`
  21. }
  22. // Metadata contains things like name and created at
  23. type Metadata struct {
  24. Name string `json:"name"`
  25. CreatedAt time.Time `json:"creationTimestamp"`
  26. Labels map[string]string `json:"labels"`
  27. Annotations map[string]string `json:"annotations"`
  28. }
  29. // Status parses information from the status key
  30. type Status struct {
  31. Info Info `json:"Info"`
  32. NodeAddresses []Address `json:"addresses"`
  33. Conditions []Condition `json:"conditions"`
  34. }
  35. // Address contains an address and a type
  36. type Address struct {
  37. Address string `json:"address"`
  38. Type string `json:"type"`
  39. }
  40. // Info contains information like what version the kubelet is running
  41. type Info struct {
  42. ContainerRuntimeVersion string `json:"containerRuntimeVersion"`
  43. KubeProxyVersion string `json:"kubeProxyVersion"`
  44. KubeletProxyVersion string `json:"kubeletVersion"`
  45. OperatingSystem string `json:"operatingSystem"`
  46. }
  47. // Condition contains various status information
  48. type Condition struct {
  49. LastHeartbeatTime time.Time `json:"lastHeartbeatTime"`
  50. LastTransitionTime time.Time `json:"lastTransitionTime"`
  51. Message string `json:"message"`
  52. Reason string `json:"reason"`
  53. Status string `json:"status"`
  54. Type string `json:"type"`
  55. }
  56. // List is used to parse out Nodes from a list
  57. type List struct {
  58. Nodes []Node `json:"items"`
  59. }
  60. // AreAllReady returns a bool depending on cluster state
  61. func AreAllReady(nodeCount int) bool {
  62. list, _ := Get()
  63. if list != nil && len(list.Nodes) == nodeCount {
  64. for _, node := range list.Nodes {
  65. for _, condition := range node.Status.Conditions {
  66. if condition.Type == "KubeletReady" && condition.Status == "false" {
  67. return false
  68. }
  69. }
  70. }
  71. return true
  72. }
  73. return false
  74. }
  75. // WaitOnReady will block until all nodes are in ready state
  76. func WaitOnReady(nodeCount int, sleep, duration time.Duration) bool {
  77. readyCh := make(chan bool, 1)
  78. errCh := make(chan error)
  79. ctx, cancel := context.WithTimeout(context.Background(), duration)
  80. defer cancel()
  81. go func() {
  82. for {
  83. select {
  84. case <-ctx.Done():
  85. errCh <- fmt.Errorf("Timeout exceeded (%s) while waiting for Nodes to become ready", duration.String())
  86. default:
  87. if AreAllReady(nodeCount) {
  88. readyCh <- true
  89. }
  90. time.Sleep(sleep)
  91. }
  92. }
  93. }()
  94. for {
  95. select {
  96. case <-errCh:
  97. return false
  98. case ready := <-readyCh:
  99. return ready
  100. }
  101. }
  102. }
  103. // Get returns the current nodes for a given kubeconfig
  104. func Get() (*List, error) {
  105. cmd := exec.Command("kubectl", "get", "nodes", "-o", "json")
  106. util.PrintCommand(cmd)
  107. out, err := cmd.CombinedOutput()
  108. if err != nil {
  109. log.Printf("Error trying to run 'kubectl get nodes':%s", string(out))
  110. return nil, err
  111. }
  112. nl := List{}
  113. err = json.Unmarshal(out, &nl)
  114. if err != nil {
  115. log.Printf("Error unmarshalling nodes json:%s", err)
  116. }
  117. return &nl, nil
  118. }
  119. // Version get the version of the server
  120. func Version() (string, error) {
  121. cmd := exec.Command("kubectl", "version", "--short")
  122. util.PrintCommand(cmd)
  123. out, err := cmd.CombinedOutput()
  124. if err != nil {
  125. log.Printf("Error trying to run 'kubectl version':%s", string(out))
  126. return "", err
  127. }
  128. split := strings.Split(string(out), "\n")
  129. exp, err := regexp.Compile(ServerVersion)
  130. if err != nil {
  131. log.Printf("Error while compiling regexp:%s", ServerVersion)
  132. }
  133. s := exp.FindStringSubmatch(split[1])
  134. return s[2], nil
  135. }
  136. // GetAddressByType will return the Address object for a given Kubernetes node
  137. func (ns *Status) GetAddressByType(t string) *Address {
  138. for _, a := range ns.NodeAddresses {
  139. if a.Type == t {
  140. return &a
  141. }
  142. }
  143. return nil
  144. }
  145. // GetByPrefix will return a []Node of all nodes that have a name that match the prefix
  146. func GetByPrefix(prefix string) ([]Node, error) {
  147. list, err := Get()
  148. if err != nil {
  149. return nil, err
  150. }
  151. nodes := make([]Node, 0)
  152. for _, n := range list.Nodes {
  153. exp, err := regexp.Compile(prefix)
  154. if err != nil {
  155. return nil, err
  156. }
  157. if exp.MatchString(n.Metadata.Name) {
  158. nodes = append(nodes, n)
  159. }
  160. }
  161. return nodes, nil
  162. }