apihandler.go 10.8 KB
Newer Older
B
bryk 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
// Copyright 2015 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package main

import (
18
	"fmt"
M
Marcin Maciaszczyk 已提交
19
	"log"
B
bryk 已提交
20
	"net/http"
M
Marcin Maciaszczyk 已提交
21
	"strconv"
B
bryk 已提交
22 23

	restful "github.com/emicklei/go-restful"
B
bryk 已提交
24
	client "k8s.io/kubernetes/pkg/client/unversioned"
25
	unversioned "k8s.io/kubernetes/pkg/client/unversioned"
B
bryk 已提交
26 27
)

28
const (
29
	RequestLogString  = "Incoming %s %s %s request from %s"
30 31 32 33 34
	ResponseLogString = "Outcoming response to %s with %d status code"
)

// Web-service filter function used for request and response logging.
func wsLogger(req *restful.Request, resp *restful.Response, chain *restful.FilterChain) {
M
Marcin Maciaszczyk 已提交
35
	log.Printf(FormatRequestLog(req))
36
	chain.ProcessFilter(req, resp)
M
Marcin Maciaszczyk 已提交
37
	log.Printf(FormatResponseLog(resp, req))
38 39 40
}

// Formats request log string.
41
// TODO(maciaszczykm): Display request body.
M
Marcin Maciaszczyk 已提交
42
func FormatRequestLog(req *restful.Request) string {
43 44 45 46 47 48
	reqURI := ""
	if req.Request.URL != nil {
		reqURI = req.Request.URL.RequestURI()
	}

	return fmt.Sprintf(RequestLogString, req.Request.Proto, req.Request.Method,
M
Marcin Maciaszczyk 已提交
49
		reqURI, req.Request.RemoteAddr)
50 51 52
}

// Formats response log string.
53
// TODO(maciaszczykm): Display response content.
M
Marcin Maciaszczyk 已提交
54 55
func FormatResponseLog(resp *restful.Response, req *restful.Request) string {
	return fmt.Sprintf(ResponseLogString, req.Request.RemoteAddr, resp.StatusCode())
56 57
}

B
bryk 已提交
58
// Creates a new HTTP handler that handles all requests to the API of the backend.
59 60
func CreateHttpApiHandler(client *client.Client, heapsterClient *unversioned.RESTClient) http.Handler {
	apiHandler := ApiHandler{client, heapsterClient}
B
bryk 已提交
61 62
	wsContainer := restful.NewContainer()

63
	deployWs := new(restful.WebService)
64
	deployWs.Filter(wsLogger)
B
bryk 已提交
65
	deployWs.Path("/api/appdeployments").
B
bryk 已提交
66 67
		Consumes(restful.MIME_JSON).
		Produces(restful.MIME_JSON)
68 69 70
	deployWs.Route(
		deployWs.POST("").
			To(apiHandler.handleDeploy).
71 72
			Reads(AppDeploymentSpec{}).
			Writes(AppDeploymentSpec{}))
B
bryk 已提交
73 74 75 76 77
	deployWs.Route(
		deployWs.POST("/validate/name").
			To(apiHandler.handleNameValidity).
			Reads(AppNameValiditySpec{}).
			Writes(AppNameValidity{}))
78 79 80 81 82 83 84 85 86
	deployWs.Route(
		deployWs.POST("/validate/protocol").
			To(apiHandler.handleProtocolValidity).
			Reads(ProtocolValiditySpec{}).
			Writes(ProtocolValidity{}))
	deployWs.Route(
		deployWs.GET("/protocols").
			To(apiHandler.handleGetAvailableProcotols).
			Writes(Protocols{}))
87 88
	wsContainer.Add(deployWs)

89
	replicaSetWs := new(restful.WebService)
90
	replicaSetWs.Filter(wsLogger)
91
	replicaSetWs.Path("/api/replicasets").
B
bryk 已提交
92
		Consumes(restful.MIME_JSON).
93
		Produces(restful.MIME_JSON)
94 95
	replicaSetWs.Route(
		replicaSetWs.GET("").
96 97
			To(apiHandler.handleGetReplicaSetList).
			Writes(ReplicaSetList{}))
98 99 100 101
	replicaSetWs.Route(
		replicaSetWs.GET("/{namespace}/{replicaSet}").
			To(apiHandler.handleGetReplicaSetDetail).
			Writes(ReplicaSetDetail{}))
102 103 104 105
	replicaSetWs.Route(
		replicaSetWs.POST("/{namespace}/{replicaSet}/update/pods").
			To(apiHandler.handleUpdateReplicasCount).
			Reads(ReplicaSetSpec{}))
106 107 108
	replicaSetWs.Route(
		replicaSetWs.DELETE("/{namespace}/{replicaSet}").
			To(apiHandler.handleDeleteReplicaSet))
109 110 111 112
	replicaSetWs.Route(
		replicaSetWs.GET("/pods/{namespace}/{replicaSet}").
			To(apiHandler.handleGetReplicaSetPods).
			Writes(ReplicaSetPods{}))
113
	wsContainer.Add(replicaSetWs)
B
bryk 已提交
114

115
	namespacesWs := new(restful.WebService)
116
	namespacesWs.Filter(wsLogger)
117 118
	namespacesWs.Path("/api/namespaces").
		Consumes(restful.MIME_JSON).
119
		Produces(restful.MIME_JSON)
120 121 122 123 124 125 126 127
	namespacesWs.Route(
		namespacesWs.POST("").
			To(apiHandler.handleCreateNamespace).
			Reads(NamespaceSpec{}).
			Writes(NamespaceSpec{}))
	namespacesWs.Route(
		namespacesWs.GET("").
			To(apiHandler.handleGetNamespaces).
128
			Writes(NamespaceList{}))
129
	wsContainer.Add(namespacesWs)
130

L
Lukasz Zajaczkowski 已提交
131
	logsWs := new(restful.WebService)
132
	logsWs.Filter(wsLogger)
L
Lukasz Zajaczkowski 已提交
133 134
	logsWs.Path("/api/logs").
		Produces(restful.MIME_JSON)
B
bryk 已提交
135 136 137 138
	logsWs.Route(
		logsWs.GET("/{namespace}/{podId}").
			To(apiHandler.handleLogs).
			Writes(Logs{}))
L
Lukasz Zajaczkowski 已提交
139
	logsWs.Route(
140
		logsWs.GET("/{namespace}/{podId}/{container}").
L
Lukasz Zajaczkowski 已提交
141 142 143 144
			To(apiHandler.handleLogs).
			Writes(Logs{}))
	wsContainer.Add(logsWs)

145
	eventsWs := new(restful.WebService)
146
	eventsWs.Filter(wsLogger)
147 148 149
	eventsWs.Path("/api/events").
		Produces(restful.MIME_JSON)
	eventsWs.Route(
150
		eventsWs.GET("/{namespace}/{replicaSet}").
151 152 153 154
			To(apiHandler.handleEvents).
			Writes(Events{}))
	wsContainer.Add(eventsWs)

B
bryk 已提交
155 156
	return wsContainer
}
B
bryk 已提交
157

158
type ApiHandler struct {
159 160
	client         *client.Client
	heapsterClient *unversioned.RESTClient
161 162 163 164
}

// Handles deploy API call.
func (apiHandler *ApiHandler) handleDeploy(request *restful.Request, response *restful.Response) {
165 166
	appDeploymentSpec := new(AppDeploymentSpec)
	if err := request.ReadEntity(appDeploymentSpec); err != nil {
167 168 169
		handleInternalError(response, err)
		return
	}
170
	if err := DeployApp(appDeploymentSpec, apiHandler.client); err != nil {
171 172 173 174
		handleInternalError(response, err)
		return
	}

175
	response.WriteHeaderAndEntity(http.StatusCreated, appDeploymentSpec)
176 177
}

B
bryk 已提交
178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194
// Handles app name validation API call.
func (apiHandler *ApiHandler) handleNameValidity(request *restful.Request, response *restful.Response) {
	spec := new(AppNameValiditySpec)
	if err := request.ReadEntity(spec); err != nil {
		handleInternalError(response, err)
		return
	}

	validity, err := ValidateAppName(spec, apiHandler.client)
	if err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeaderAndEntity(http.StatusCreated, validity)
}

195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210
// Handles protocol validation API call.
func (apiHandler *ApiHandler) handleProtocolValidity(request *restful.Request, response *restful.Response) {
	spec := new(ProtocolValiditySpec)
	if err := request.ReadEntity(spec); err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeaderAndEntity(http.StatusCreated, ValidateProtocol(spec))
}

// Handles get available protocols API call.
func (apiHandler *ApiHandler) handleGetAvailableProcotols(request *restful.Request, response *restful.Response) {
	response.WriteHeaderAndEntity(http.StatusCreated, GetAvailableProtocols())
}

211 212
// Handles get Replica Set list API call.
func (apiHandler *ApiHandler) handleGetReplicaSetList(
213 214
	request *restful.Request, response *restful.Response) {

215
	result, err := GetReplicaSetList(apiHandler.client)
216 217 218 219 220 221 222 223
	if err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeaderAndEntity(http.StatusCreated, result)
}

224 225 226 227 228 229
// Handles get Replica Set detail API call.
func (apiHandler *ApiHandler) handleGetReplicaSetDetail(
	request *restful.Request, response *restful.Response) {

	namespace := request.PathParameter("namespace")
	replicaSet := request.PathParameter("replicaSet")
230
	result, err := GetReplicaSetDetail(apiHandler.client, apiHandler.heapsterClient, namespace, replicaSet)
231 232 233 234 235 236 237 238
	if err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeaderAndEntity(http.StatusCreated, result)
}

239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260
// Handles update of Replica Set pods update API call.
func (apiHandler *ApiHandler) handleUpdateReplicasCount(
	request *restful.Request, response *restful.Response) {

	namespace := request.PathParameter("namespace")
	replicaSetName := request.PathParameter("replicaSet")
	replicaSetSpec := new(ReplicaSetSpec)

	if err := request.ReadEntity(replicaSetSpec); err != nil {
		handleInternalError(response, err)
		return
	}

	if err := UpdateReplicasCount(apiHandler.client, namespace, replicaSetName,
		replicaSetSpec); err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeader(http.StatusAccepted)
}

261 262 263 264 265 266 267 268 269 270 271 272 273 274 275
// Handles delete Replica Set API call.
func (apiHandler *ApiHandler) handleDeleteReplicaSet(
	request *restful.Request, response *restful.Response) {

	namespace := request.PathParameter("namespace")
	replicaSet := request.PathParameter("replicaSet")

	if err := DeleteReplicaSetWithPods(apiHandler.client, namespace, replicaSet); err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeader(http.StatusOK)
}

276 277 278 279 280 281
// Handles get Replica Set Pods API call.
func (apiHandler *ApiHandler) handleGetReplicaSetPods(
	request *restful.Request, response *restful.Response) {

	namespace := request.PathParameter("namespace")
	replicaSet := request.PathParameter("replicaSet")
282 283 284 285 286
	limit, err := strconv.Atoi(request.QueryParameter("limit"))
	if err != nil {
		limit = 0
	}
	result, err := GetReplicaSetPods(apiHandler.client, namespace, replicaSet, limit)
287 288 289 290 291 292 293 294
	if err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeaderAndEntity(http.StatusCreated, result)
}

295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310
// Handles namespace creation API call.
func (apiHandler *ApiHandler) handleCreateNamespace(request *restful.Request,
	response *restful.Response) {
	namespaceSpec := new(NamespaceSpec)
	if err := request.ReadEntity(namespaceSpec); err != nil {
		handleInternalError(response, err)
		return
	}
	if err := CreateNamespace(namespaceSpec, apiHandler.client); err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeaderAndEntity(http.StatusCreated, namespaceSpec)
}

311
// Handles get namespace list API call.
312
func (apiHandler *ApiHandler) handleGetNamespaces(
313 314 315 316 317 318 319 320 321 322 323
	request *restful.Request, response *restful.Response) {

	result, err := GetNamespaceList(apiHandler.client)
	if err != nil {
		handleInternalError(response, err)
		return
	}

	response.WriteHeaderAndEntity(http.StatusCreated, result)
}

L
Lukasz Zajaczkowski 已提交
324 325 326 327
// Handles log API call.
func (apiHandler *ApiHandler) handleLogs(request *restful.Request, response *restful.Response) {
	namespace := request.PathParameter("namespace")
	podId := request.PathParameter("podId")
328
	container := request.PathParameter("container")
B
bryk 已提交
329 330 331 332 333
	var containerPtr *string = nil
	if len(container) > 0 {
		containerPtr = &container
	}
	result, err := GetPodLogs(apiHandler.client, namespace, podId, containerPtr)
L
Lukasz Zajaczkowski 已提交
334 335 336 337 338 339 340
	if err != nil {
		handleInternalError(response, err)
		return
	}
	response.WriteHeaderAndEntity(http.StatusCreated, result)
}

341 342 343
// Handles event API call.
func (apiHandler *ApiHandler) handleEvents(request *restful.Request, response *restful.Response) {
	namespace := request.PathParameter("namespace")
344 345
	replicaSet := request.PathParameter("replicaSet")
	result, err := GetEvents(apiHandler.client, namespace, replicaSet)
346 347 348 349 350 351 352
	if err != nil {
		handleInternalError(response, err)
		return
	}
	response.WriteHeaderAndEntity(http.StatusCreated, result)
}

B
bryk 已提交
353
// Handler that writes the given error to the response and sets appropriate HTTP status headers.
354
func handleInternalError(response *restful.Response, err error) {
M
Marcin Maciaszczyk 已提交
355
	log.Print(err)
B
bryk 已提交
356 357 358
	response.AddHeader("Content-Type", "text/plain")
	response.WriteErrorString(http.StatusInternalServerError, err.Error()+"\n")
}