handlers.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300
  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 handler
  15. import (
  16. "context"
  17. "fmt"
  18. "net/http"
  19. "strings"
  20. "time"
  21. "yunion.io/x/jsonutils"
  22. "yunion.io/x/log"
  23. "yunion.io/x/pkg/errors"
  24. baremetalapi "yunion.io/x/onecloud/pkg/apis/baremetal"
  25. "yunion.io/x/onecloud/pkg/appsrv"
  26. "yunion.io/x/onecloud/pkg/baremetal"
  27. "yunion.io/x/onecloud/pkg/baremetal/options"
  28. baremetalstatus "yunion.io/x/onecloud/pkg/baremetal/status"
  29. "yunion.io/x/onecloud/pkg/baremetal/tasks"
  30. baremetaltypes "yunion.io/x/onecloud/pkg/baremetal/types"
  31. "yunion.io/x/onecloud/pkg/baremetal/utils/ipmitool"
  32. app_common "yunion.io/x/onecloud/pkg/cloudcommon/app"
  33. "yunion.io/x/onecloud/pkg/httperrors"
  34. "yunion.io/x/onecloud/pkg/mcclient/auth"
  35. "yunion.io/x/onecloud/pkg/util/redfish"
  36. )
  37. var registerWorkMan *appsrv.SWorkerManager
  38. func init() {
  39. registerWorkMan = appsrv.NewWorkerManager("bm_register_worker", 20, 1024, false)
  40. }
  41. func InitHandlers(app *appsrv.Application) {
  42. initBaremetalsHandler(app)
  43. }
  44. func AddHandler(app *appsrv.Application, method string, prefix string,
  45. handler func(context.Context, http.ResponseWriter, *http.Request)) *appsrv.SHandlerInfo {
  46. return app.AddHandler(method, prefix, auth.Authenticate(handler))
  47. }
  48. func AddHandler2(app *appsrv.Application, method string, prefix string,
  49. handler func(context.Context, http.ResponseWriter, *http.Request),
  50. metadata map[string]interface{}, name string, tags map[string]string) *appsrv.SHandlerInfo {
  51. return app.AddHandler2(method, prefix, auth.Authenticate(handler), metadata, name, tags)
  52. }
  53. func customizeHandlerInfo(info *appsrv.SHandlerInfo) {
  54. if info.GetName(nil) == "baremetal-register" {
  55. info.SetProcessTimeout(time.Second * 300).SetWorkerManager(registerWorkMan)
  56. }
  57. }
  58. func initBaremetalsHandler(app *appsrv.Application) {
  59. app_common.ExportOptionsHandler(app, &options.Options)
  60. // baremetal actions handler
  61. AddHandler(app, "GET", bmActionPrefix("notify"), bmObjMiddleware(handleBaremetalNotify))
  62. AddHandler(app, "POST", bmActionPrefix("maintenance"), bmObjMiddleware(handleBaremetalMaintenance))
  63. AddHandler(app, "POST", bmActionPrefix("unmaintenance"), bmObjMiddleware(handleBaremetalUnmaintenance))
  64. AddHandler(app, "POST", bmActionPrefix("delete"), bmObjMiddlewareWithFetch(handleBaremetalDelete, false))
  65. AddHandler(app, "POST", bmActionPrefix("syncstatus"), bmObjMiddleware(handleBaremetalSyncStatus))
  66. AddHandler(app, "POST", bmActionPrefix("sync-config"), bmObjMiddleware(handleBaremetalSyncConfig))
  67. AddHandler(app, "POST", bmActionPrefix("sync-ipmi"), bmObjMiddleware(handleBaremetalSyncIPMI))
  68. AddHandler(app, "POST", bmActionPrefix("prepare"), bmObjMiddleware(handleBaremetalPrepare))
  69. AddHandler(app, "POST", bmActionPrefix("reset-bmc"), bmObjMiddleware(handleBaremetalResetBMC))
  70. AddHandler(app, "POST", bmActionPrefix("ipmi-probe"), bmObjMiddleware(handleBaremetalIpmiProbe))
  71. AddHandler(app, "POST", bmActionPrefix("cdrom"), bmObjMiddleware(handleBaremetalCdromTask))
  72. AddHandler(app, "POST", bmActionPrefix("jnlp"), bmObjMiddleware(handleBaremetalJnlpTask))
  73. AddHandler(app, "POST", "/baremetals/validate-ipmi", handleBaremetalValidateIPMI())
  74. // server actions handler
  75. AddHandler(app, "POST", srvActionPrefix("create"), srvClassMiddleware(handleServerCreate))
  76. AddHandler(app, "POST", srvActionPrefix("deploy"), srvObjMiddleware(handleServerDeploy))
  77. AddHandler(app, "POST", srvActionPrefix("rebuild"), srvObjMiddleware(handleServerRebuild))
  78. AddHandler(app, "POST", srvActionPrefix("start"), srvObjMiddleware(handleServerStart))
  79. AddHandler(app, "POST", srvActionPrefix("stop"), srvObjMiddleware(handleServerStop))
  80. AddHandler(app, "POST", srvActionPrefix("reset"), srvObjMiddleware(handleServerReset))
  81. AddHandler(app, "POST", srvActionPrefix("status"), srvObjMiddleware(handleServerStatus))
  82. AddHandler(app, "DELETE", srvIdPrefix(), srvObjMiddleware(handleServerDelete))
  83. handInfo := AddHandler2(app, "POST", bmRegisterPrefix(),
  84. bmRegisterMiddleware(handleBaremetalRegister), nil, "baremetal-register", nil)
  85. customizeHandlerInfo(handInfo)
  86. }
  87. func handleBaremetalNotify(ctx *Context, bm *baremetal.SBaremetalInstance) {
  88. key, err := ctx.Query().GetString("key")
  89. if err != nil {
  90. ctx.ResponseError(httperrors.NewInputParameterError("Not found key in query"))
  91. return
  92. }
  93. remoteAddr := ctx.RequestRemoteIP()
  94. err = bm.SaveSSHConfig(remoteAddr, key)
  95. if err != nil {
  96. log.Errorf("Save baremetal %s ssh config: %v", bm.GetId(), err)
  97. }
  98. // execute BaremetalServerPrepareTask
  99. task := bm.GetTask()
  100. if task == nil {
  101. task = tasks.NewBaremetalServerPrepareTask(bm)
  102. bm.SyncStatus(ctx, baremetalstatus.PREPARE, "")
  103. }
  104. log.Infof("Get notify from pxe rom os, start exec task: %s", task.GetName())
  105. task.SSHExecute(remoteAddr, key, nil)
  106. ctx.ResponseOk()
  107. }
  108. func handleBaremetalMaintenance(ctx *Context, bm *baremetal.SBaremetalInstance) {
  109. bm.StartBaremetalMaintenanceTask(ctx.UserCred(), ctx.TaskId(), ctx.Data())
  110. ctx.ResponseOk()
  111. }
  112. func handleBaremetalUnmaintenance(ctx *Context, bm *baremetal.SBaremetalInstance) {
  113. bm.StartBaremetalUnmaintenanceTask(ctx.UserCred(), ctx.TaskId(), ctx.Data())
  114. ctx.ResponseOk()
  115. }
  116. func handleBaremetalDelete(ctx *Context, bm *baremetal.SBaremetalInstance) {
  117. ctx.DelayProcess(bm.DelayedRemove, nil)
  118. ctx.ResponseOk()
  119. }
  120. func handleBaremetalSyncStatus(ctx *Context, bm *baremetal.SBaremetalInstance) {
  121. ctx.DelayProcess(bm.DelayedSyncStatus, nil)
  122. ctx.ResponseOk()
  123. }
  124. func handleBaremetalSyncConfig(ctx *Context, bm *baremetal.SBaremetalInstance) {
  125. ctx.DelayProcess(bm.DelayedSyncDesc, nil)
  126. ctx.ResponseOk()
  127. }
  128. func handleBaremetalSyncIPMI(ctx *Context, bm *baremetal.SBaremetalInstance) {
  129. ctx.DelayProcess(bm.DelayedSyncIPMIInfo, nil)
  130. ctx.ResponseOk()
  131. }
  132. func handleBaremetalPrepare(ctx *Context, bm *baremetal.SBaremetalInstance) {
  133. bm.StartBaremetalReprepareTask(ctx.UserCred(), ctx.TaskId(), ctx.Data())
  134. ctx.ResponseOk()
  135. }
  136. func handleBaremetalResetBMC(ctx *Context, bm *baremetal.SBaremetalInstance) {
  137. bm.StartBaremetalResetBMCTask(ctx.UserCred(), ctx.TaskId(), ctx.Data())
  138. ctx.ResponseOk()
  139. }
  140. func handleBaremetalIpmiProbe(ctx *Context, bm *baremetal.SBaremetalInstance) {
  141. bm.StartBaremetalIpmiProbeTask(ctx, ctx.UserCred(), ctx.TaskId(), ctx.Data())
  142. ctx.ResponseOk()
  143. }
  144. func handleBaremetalCdromTask(ctx *Context, bm *baremetal.SBaremetalInstance) {
  145. bm.StartBaremetalCdromTask(ctx.UserCred(), ctx.TaskId(), ctx.Data())
  146. ctx.ResponseOk()
  147. }
  148. func handleBaremetalJnlpTask(ctx *Context, bm *baremetal.SBaremetalInstance) {
  149. jnlp, err := bm.GetConsoleJNLP(ctx)
  150. if err != nil {
  151. ctx.ResponseError(errors.Wrap(err, "GetConsoleJNLP"))
  152. return
  153. }
  154. result := jsonutils.NewDict()
  155. result.Add(jsonutils.NewString(jnlp), "jnlp")
  156. ctx.ResponseJson(result)
  157. }
  158. func handleBaremetalValidateIPMI() appsrv.FilterHandler {
  159. return auth.Authenticate(func(ctx context.Context, w http.ResponseWriter, r *http.Request) {
  160. _, _, body := appsrv.FetchEnv(ctx, w, r)
  161. if body == nil {
  162. httperrors.BadRequestError(ctx, w, "body is empty")
  163. return
  164. }
  165. validate := func() (*baremetalapi.ValidateIPMIResponse, error) {
  166. input := new(baremetalapi.ValidateIPMIRequest)
  167. if err := body.Unmarshal(input); err != nil {
  168. return nil, errors.Wrapf(err, "unmarshal validate ipmi request: %s", body)
  169. }
  170. var endpoint = "https://" + input.Ip
  171. if strings.Contains(input.Ip, ":") {
  172. endpoint = fmt.Sprintf("https://[%s]", input.Ip)
  173. }
  174. redfishCli := redfish.NewRedfishDriver(ctx, endpoint, input.Username, input.Password, false)
  175. resp := &baremetalapi.ValidateIPMIResponse{}
  176. if redfishCli == nil {
  177. resp.IsRedfishSupported = false
  178. // use ipmitool to validate
  179. tool := ipmitool.NewLanPlusIPMI(input.Ip, input.Username, input.Password)
  180. info, err := ipmitool.GetSysInfo(tool)
  181. if err != nil {
  182. return nil, errors.Wrap(err, "GetSysInfo by ipmitool")
  183. }
  184. resp.IPMISystemInfo = info
  185. } else {
  186. path, info, err := redfishCli.GetSystemInfo(ctx)
  187. if err != nil {
  188. return nil, errors.Wrapf(err, "get system info")
  189. }
  190. resp.RedfishSystemInfo = &baremetalapi.RedfishSystemInfo{
  191. Path: path,
  192. Info: info,
  193. }
  194. }
  195. return resp, nil
  196. }
  197. resp, err := validate()
  198. if err != nil {
  199. httperrors.GeneralServerError(ctx, w, err)
  200. return
  201. }
  202. appsrv.SendStruct(w, resp)
  203. })
  204. }
  205. func handleServerCreate(ctx *Context, bm *baremetal.SBaremetalInstance) {
  206. err := bm.StartServerCreateTask(ctx, ctx.UserCred(), ctx.TaskId(), ctx.Data())
  207. if err != nil {
  208. ctx.ResponseError(httperrors.NewGeneralError(err))
  209. return
  210. }
  211. ctx.ResponseOk()
  212. }
  213. func handleServerDelete(ctx *Context, bm *baremetal.SBaremetalInstance, _ baremetaltypes.IBaremetalServer) {
  214. bm.StartServerDestroyTask(ctx.UserCred(), ctx.TaskId(), nil)
  215. ctx.ResponseOk()
  216. }
  217. func handleServerDeploy(ctx *Context, bm *baremetal.SBaremetalInstance, _ baremetaltypes.IBaremetalServer) {
  218. if err := bm.StartServerDeployTask(ctx.UserCred(), ctx.TaskId(), ctx.Data()); err != nil {
  219. ctx.ResponseError(httperrors.NewGeneralError(err))
  220. return
  221. }
  222. ctx.ResponseOk()
  223. }
  224. func handleServerRebuild(ctx *Context, bm *baremetal.SBaremetalInstance, _ baremetaltypes.IBaremetalServer) {
  225. if err := bm.StartServerRebuildTask(ctx.UserCred(), ctx.TaskId(), ctx.Data()); err != nil {
  226. ctx.ResponseError(httperrors.NewGeneralError(err))
  227. return
  228. }
  229. ctx.ResponseOk()
  230. }
  231. func handleServerStart(ctx *Context, bm *baremetal.SBaremetalInstance, _ baremetaltypes.IBaremetalServer) {
  232. if err := bm.StartServerStartTask(ctx.UserCred(), ctx.TaskId(), ctx.Data()); err != nil {
  233. ctx.ResponseError(httperrors.NewGeneralError(err))
  234. return
  235. }
  236. ctx.ResponseOk()
  237. }
  238. func handleServerStop(ctx *Context, bm *baremetal.SBaremetalInstance, _ baremetaltypes.IBaremetalServer) {
  239. if bm.HasBMC() {
  240. if err := bm.StartServerStopTask(ctx.UserCred(), ctx.TaskId(), ctx.Data()); err != nil {
  241. ctx.ResponseError(httperrors.NewGeneralError(err))
  242. return
  243. }
  244. } else {
  245. bm.StartBaremetalMaintenanceTask(ctx.UserCred(), ctx.TaskId(), ctx.Data())
  246. }
  247. ctx.ResponseOk()
  248. }
  249. func handleServerReset(ctx *Context, bm *baremetal.SBaremetalInstance, _ baremetaltypes.IBaremetalServer) {
  250. ctx.DelayProcess(bm.DelayedServerReset, nil)
  251. ctx.ResponseOk()
  252. }
  253. func handleServerStatus(ctx *Context, bm *baremetal.SBaremetalInstance, _ baremetaltypes.IBaremetalServer) {
  254. ctx.DelayProcess(bm.DelayedServerStatus, nil)
  255. ctx.ResponseOk()
  256. }
  257. func handleBaremetalRegister(ctx *Context, input *baremetal.BmRegisterInput) {
  258. ctx.DelayProcess(func(_ context.Context, data jsonutils.JSONObject) (jsonutils.JSONObject, error) {
  259. baremetal.GetBaremetalManager().RegisterBaremetal(ctx, ctx.userCred, input)
  260. return nil, nil
  261. }, nil)
  262. }