clusters.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. // Copyright 2019 Yunion
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package k8s
  15. import (
  16. "fmt"
  17. "io/ioutil"
  18. "os"
  19. "path/filepath"
  20. "yunion.io/x/jsonutils"
  21. "yunion.io/x/log"
  22. "yunion.io/x/pkg/errors"
  23. "yunion.io/x/pkg/util/shellutils"
  24. "yunion.io/x/onecloud/cmd/climc/shell"
  25. "yunion.io/x/onecloud/pkg/mcclient"
  26. "yunion.io/x/onecloud/pkg/mcclient/modules/k8s"
  27. o "yunion.io/x/onecloud/pkg/mcclient/options/k8s"
  28. )
  29. func initKubeCluster() {
  30. cmdN := func(action string) string {
  31. return fmt.Sprintf("k8s-cluster-%s", action)
  32. }
  33. cmd := NewK8sResourceCmd(k8s.KubeClusters)
  34. cmd.SetKeyword("cluster")
  35. cmd.ShowEvent()
  36. cmd.List(new(o.ClusterListOptions))
  37. cmd.Show(new(o.IdentOptions))
  38. cmd.Create(new(o.K8SClusterCreateOptions))
  39. cmd.Perform("sync", new(o.ClusterSyncOptions))
  40. cmd.Perform("syncstatus", new(o.IdentOptions))
  41. cmd.Perform("deploy", new(o.ClusterDeployOptions))
  42. cmd.Get("components-status", new(o.IdentOptions))
  43. cmd.Get("api-resources", new(o.IdentOptions))
  44. cmd.Get("cluster-users", new(o.IdentOptions))
  45. cmd.Get("cluster-user-groups", new(o.IdentOptions))
  46. cmd.Perform("purge", new(o.ClusterPurgeOptions))
  47. cmd.Perform("delete-machines", new(o.KubeClusterDeleteMachinesOptions))
  48. cmd.Perform("add-machines", new(o.KubeClusterAddMachinesOptions))
  49. cmd.PerformClass("gc", new(o.ClusterGCOpts))
  50. cmd.Perform("set-extra-config", new(o.ClusterSetExtraConfigOpt))
  51. cmd.Get("extra-config", new(o.IdentOptions))
  52. R(&o.KubeClusterImportOptions{}, cmdN("import"), "Import k8s cluster", func(s *mcclient.ClientSession, args *o.KubeClusterImportOptions) error {
  53. params, err := args.Params()
  54. if err != nil {
  55. return err
  56. }
  57. cluster, err := k8s.KubeClusters.Create(s, params)
  58. if err != nil {
  59. return err
  60. }
  61. printObject(cluster)
  62. return nil
  63. })
  64. R(&o.ClusterDeleteOptions{}, cmdN("delete"), "Delete cluster", func(s *mcclient.ClientSession, args *o.ClusterDeleteOptions) error {
  65. ret := k8s.KubeClusters.BatchDeleteWithParam(s, args.ID, nil, nil)
  66. printBatchResults(ret, k8s.KubeClusters.GetColumns(s))
  67. return nil
  68. })
  69. R(&o.IdentOptions{}, cmdN("kubeconfig"), "Generate kubeconfig of a cluster", func(s *mcclient.ClientSession, args *o.IdentOptions) error {
  70. ret, err := k8s.KubeClusters.GetSpecific(s, args.ID, "kubeconfig", nil)
  71. if err != nil {
  72. return err
  73. }
  74. conf, err := ret.GetString("kubeconfig")
  75. if err != nil {
  76. return err
  77. }
  78. fmt.Println(conf)
  79. return nil
  80. })
  81. type SetKubeconfigOptions struct {
  82. o.IdentOptions
  83. FILE string `json:"filepath of kubeconfig"`
  84. }
  85. R(new(SetKubeconfigOptions), cmdN("set-kubeconfig"), "Set kubeconfig of a cluster", func(s *mcclient.ClientSession, args *SetKubeconfigOptions) error {
  86. content, err := ioutil.ReadFile(args.FILE)
  87. if err != nil {
  88. return errors.Wrapf(err, "read file content of %q", args.FILE)
  89. }
  90. input := map[string]string{
  91. "kubeconfig": string(content),
  92. }
  93. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "set-kubeconfig", jsonutils.Marshal(input))
  94. if err != nil {
  95. return err
  96. }
  97. printObject(ret)
  98. return nil
  99. })
  100. R(&o.ClusterGetAddonsOpt{}, cmdN("addons"), "Get addon manifest of a cluster", func(s *mcclient.ClientSession, args *o.ClusterGetAddonsOpt) error {
  101. params, err := args.Params()
  102. if err != nil {
  103. return err
  104. }
  105. ret, err := k8s.KubeClusters.GetSpecific(s, args.ID, "addons", params)
  106. if err != nil {
  107. return err
  108. }
  109. conf, err := ret.GetString("addons")
  110. if err != nil {
  111. return err
  112. }
  113. fmt.Println(conf)
  114. return nil
  115. })
  116. R(&o.ClusterK8sVersions{}, cmdN("k8s-versions"), "Get kubernetes deployable versions", func(s *mcclient.ClientSession, args *o.ClusterK8sVersions) error {
  117. params := jsonutils.NewDict()
  118. params.Add(jsonutils.NewString(args.PROVIDER), "provider")
  119. ret, err := k8s.KubeClusters.Get(s, "k8s-versions", params)
  120. if err != nil {
  121. return err
  122. }
  123. fmt.Println(ret.String())
  124. return nil
  125. })
  126. R(&o.ClusterK8sVersions{}, cmdN("usable-instances"), "Get deploy usable instance", func(s *mcclient.ClientSession, args *o.ClusterK8sVersions) error {
  127. params := jsonutils.NewDict()
  128. params.Add(jsonutils.NewString(args.PROVIDER), "provider")
  129. ret, err := k8s.KubeClusters.Get(s, "usable-instances", params)
  130. if err != nil {
  131. return err
  132. }
  133. fmt.Println(ret.String())
  134. return nil
  135. })
  136. R(&o.ClusterCheckOptions{}, cmdN("check-system-ready"), "Check system cluster status", func(s *mcclient.ClientSession, args *o.ClusterCheckOptions) error {
  137. ret, err := k8s.KubeClusters.PerformClassAction(s, "check-system-ready", nil)
  138. if err != nil {
  139. return err
  140. }
  141. fmt.Println(ret.String())
  142. return nil
  143. })
  144. R(&o.IdentOptions{}, cmdN("apply-addons"), "Apply base requirements addons", func(s *mcclient.ClientSession, args *o.IdentOptions) error {
  145. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "apply-addons", nil)
  146. if err != nil {
  147. return err
  148. }
  149. printObject(ret)
  150. return nil
  151. })
  152. getComponentSetting := func(s *mcclient.ClientSession, args *o.ClusterComponentType, asHelmValues bool) (jsonutils.JSONObject, error) {
  153. q := jsonutils.NewDict()
  154. q.Add(jsonutils.NewString(args.TYPE), "type")
  155. q.Add(jsonutils.JSONTrue, "system")
  156. q.Add(jsonutils.NewBool(asHelmValues), "as_helm_values")
  157. return k8s.KubeClusters.GetSpecific(s, args.ID, "component-setting", q)
  158. }
  159. R(&o.ClusterComponentTypeOptions{}, cmdN("component-setting"), "Get cluster component setting", func(s *mcclient.ClientSession, args *o.ClusterComponentTypeOptions) error {
  160. ret, err := getComponentSetting(s, &args.ClusterComponentType, args.AsHelmValues)
  161. if err != nil {
  162. return err
  163. }
  164. if args.AsHelmValues {
  165. printObjectYAML(ret)
  166. } else {
  167. printObject(ret)
  168. }
  169. return nil
  170. })
  171. R(&o.ClusterEnableComponentCephCSIOpt{}, cmdN("component-enable-ceph-csi"), "Enable cluster ceph csi component", func(s *mcclient.ClientSession, args *o.ClusterEnableComponentCephCSIOpt) error {
  172. params, err := args.Params()
  173. if err != nil {
  174. return err
  175. }
  176. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "enable-component", params)
  177. if err != nil {
  178. return err
  179. }
  180. printObject(ret)
  181. return nil
  182. })
  183. R(&o.ClusterEnableComponentMonitorOpt{}, cmdN("component-enable-monitor"), "Enable cluster monitor component", func(s *mcclient.ClientSession, args *o.ClusterEnableComponentMonitorOpt) error {
  184. params, err := args.Params()
  185. if err != nil {
  186. return err
  187. }
  188. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "enable-component", params)
  189. if err != nil {
  190. return err
  191. }
  192. printObject(ret)
  193. return nil
  194. })
  195. R(&o.ClusterEnableComponentFluentBitOpt{}, cmdN("component-enable-fluentbit"), "Enable cluster fluentbit component", func(s *mcclient.ClientSession, args *o.ClusterEnableComponentFluentBitOpt) error {
  196. params, err := args.Params()
  197. if err != nil {
  198. return err
  199. }
  200. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "enable-component", params)
  201. if err != nil {
  202. return err
  203. }
  204. printObject(ret)
  205. return nil
  206. })
  207. rEnableMinio := func(opt shell.IPerformOpt, cmd string, desc string) {
  208. R(opt, cmd, desc, func(s *mcclient.ClientSession, args shell.IPerformOpt) error {
  209. params, err := args.Params()
  210. if err != nil {
  211. return err
  212. }
  213. ret, err := k8s.KubeClusters.PerformAction(s, args.GetId(), "enable-component", params)
  214. if err != nil {
  215. return err
  216. }
  217. printObject(ret)
  218. return nil
  219. })
  220. }
  221. rEnableMinio(
  222. new(o.ClusterEnableComponentMinioOpt),
  223. cmdN("component-enable-minio"),
  224. "Enable cluster minio component",
  225. )
  226. rEnableMinio(
  227. new(o.ClusterEnableComponentMonitorMinioOpt),
  228. cmdN("component-enable-monitor-minio"),
  229. "Enable cluster monitor stack's minio component",
  230. )
  231. R(&o.ClusterEnableComponentThanosOpt{}, cmdN("component-enable-thanos"), "Enable cluster thanos component", func(s *mcclient.ClientSession, args *o.ClusterEnableComponentThanosOpt) error {
  232. params, err := args.Params()
  233. if err != nil {
  234. return err
  235. }
  236. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "enable-component", params)
  237. if err != nil {
  238. return err
  239. }
  240. printObject(ret)
  241. return nil
  242. })
  243. R(&o.ClusterDisableComponent{}, cmdN("component-disable"), "Enable cluster component", func(s *mcclient.ClientSession, args *o.ClusterDisableComponent) error {
  244. params := args.Params()
  245. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "disable-component", params)
  246. if err != nil {
  247. return err
  248. }
  249. printObject(ret)
  250. return nil
  251. })
  252. R(&o.ClusterDisableComponent{}, cmdN("component-delete"), "Delete cluster component", func(s *mcclient.ClientSession, args *o.ClusterDisableComponent) error {
  253. params := args.Params()
  254. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "delete-component", params)
  255. if err != nil {
  256. return err
  257. }
  258. printObject(ret)
  259. return nil
  260. })
  261. R(&o.ClusterUpdateComponentCephCSIOpt{}, cmdN("component-update-ceph-csi"), "Update cluster component ceph csi", func(s *mcclient.ClientSession, args *o.ClusterUpdateComponentCephCSIOpt) error {
  262. params, err := args.Params()
  263. if err != nil {
  264. return err
  265. }
  266. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "update-component", params)
  267. if err != nil {
  268. return err
  269. }
  270. printObject(ret)
  271. return nil
  272. })
  273. type ClusterComponentTypeUpdate struct {
  274. o.ClusterComponentType
  275. Force bool `help:"Force update"`
  276. }
  277. R(&ClusterComponentTypeUpdate{}, cmdN("component-update"), "Update cluster component", func(s *mcclient.ClientSession, args *ClusterComponentTypeUpdate) error {
  278. // 1. get current setting
  279. setting, err := getComponentSetting(s, &args.ClusterComponentType, false)
  280. if err != nil {
  281. return errors.Wrap(err, "get component setting")
  282. }
  283. // 2. edit yaml
  284. yaml, err := shellutils.Edit(setting.YAMLString())
  285. if len(yaml) == 0 {
  286. if !args.Force {
  287. log.Infof("Nothing to update")
  288. return nil
  289. }
  290. yaml = setting.YAMLString()
  291. }
  292. nowSetting, err := jsonutils.ParseYAML(yaml)
  293. if err != nil {
  294. return err
  295. }
  296. params := args.Params(args.TYPE)
  297. params.Update(nowSetting)
  298. params.Add(jsonutils.NewBool(args.Force), "force")
  299. // 3. call update api
  300. ret, err := k8s.KubeClusters.PerformAction(s, args.ID, "update-component", params)
  301. if err != nil {
  302. return err
  303. }
  304. printObject(ret)
  305. return nil
  306. })
  307. type GetKubesprayConfigOpt struct {
  308. o.IdentOptions
  309. OUTPUT string `help:"Output directory to store config files"`
  310. }
  311. R(&GetKubesprayConfigOpt{}, cmdN("kubespray-config"), "Get cluster kubespray config", func(s *mcclient.ClientSession, args *GetKubesprayConfigOpt) error {
  312. conf, err := k8s.KubeClusters.GetSpecific(s, args.ID, "kubespray-config", nil)
  313. if err != nil {
  314. return err
  315. }
  316. inventoryContent, err := conf.GetString("inventory_content")
  317. if err != nil {
  318. return errors.Wrap(err, "get inventory content")
  319. }
  320. vars, err := conf.Get("vars")
  321. if err != nil {
  322. return errors.Wrap(err, "get variables")
  323. }
  324. privateKey, err := conf.GetString("private_key")
  325. if err != nil {
  326. return errors.Wrap(err, "get private key")
  327. }
  328. if err := os.MkdirAll(args.OUTPUT, 0755); err != nil {
  329. return errors.Wrap(err, "mkdir")
  330. }
  331. writeFile := func(name, content string) error {
  332. fp := filepath.Join(args.OUTPUT, name)
  333. log.Infof("Write file: %s", fp)
  334. if err := ioutil.WriteFile(fp, []byte(content), 0644); err != nil {
  335. return errors.Wrap(err, "write file")
  336. }
  337. if err := os.Chmod(fp, 0600); err != nil {
  338. return errors.Wrap(err, "chmod")
  339. }
  340. return nil
  341. }
  342. iPath := filepath.Join(args.OUTPUT, "hosts.ini")
  343. vPath := filepath.Join(args.OUTPUT, "vars.json")
  344. kPath := filepath.Join(args.OUTPUT, "private_key")
  345. for fPath, content := range map[string]string{
  346. iPath: inventoryContent,
  347. vPath: vars.PrettyString(),
  348. kPath: privateKey,
  349. } {
  350. name := filepath.Base(fPath)
  351. if err := writeFile(name, content); err != nil {
  352. return errors.Wrapf(err, "write file %s", name)
  353. }
  354. }
  355. fmt.Printf("try cmd:\nansible-playbook -i %s cluster.yml -b -v --private-key %s --extra-vars @%s\n", iPath, kPath, vPath)
  356. return nil
  357. })
  358. }