| /* |
| Licensed to the Apache Software Foundation (ASF) under one or more |
| contributor license agreements. See the NOTICE file distributed with |
| this work for additional information regarding copyright ownership. |
| The ASF licenses this file to You 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 pipelines |
| |
| import ( |
| "github.com/apache/incubator-devlake/core/errors" |
| "github.com/apache/incubator-devlake/core/models" |
| "github.com/apache/incubator-devlake/server/api/shared" |
| "github.com/apache/incubator-devlake/server/services" |
| "net/http" |
| "os" |
| "path/filepath" |
| "strconv" |
| |
| "github.com/gin-gonic/gin" |
| "github.com/gin-gonic/gin/binding" |
| ) |
| |
| // @Summary Create and run a new pipeline |
| // @Description Create and run a new pipeline |
| // @Tags framework/pipelines |
| // @Accept application/json |
| // @Param pipeline body models.NewPipeline true "json" |
| // @Success 200 {object} models.Pipeline |
| // @Failure 400 {string} errcode.Error "Bad Request" |
| // @Failure 500 {string} errcode.Error "Internal Error" |
| // @Router /pipelines [post] |
| func Post(c *gin.Context) { |
| newPipeline := &models.NewPipeline{} |
| |
| err := c.MustBindWith(newPipeline, binding.JSON) |
| if err != nil { |
| shared.ApiOutputError(c, errors.BadInput.Wrap(err, "bad JSON request body format")) |
| return |
| } |
| |
| pipeline, err := services.CreatePipeline(newPipeline, true) |
| // Return all created tasks to the User |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "error creating pipeline")) |
| return |
| } |
| shared.ApiOutputSuccess(c, pipeline, http.StatusCreated) |
| } |
| |
| // @Summary Get list of pipelines |
| // @Description GET /pipelines?status=TASK_RUNNING&pending=1&label=search_text&page=1&pagesize=10 |
| // @Tags framework/pipelines |
| // @Param status query string false "status" |
| // @Param pending query int false "pending" |
| // @Param page query int false "page" |
| // @Param pagesize query int false "pagesize" |
| // @Param blueprint_id query int false "blueprint_id" |
| // @Param label query string false "label" |
| // @Success 200 {object} shared.ResponsePipelines |
| // @Failure 400 {string} errcode.Error "Bad Request" |
| // @Failure 500 {string} errcode.Error "Internal Error" |
| // @Router /pipelines [get] |
| func Index(c *gin.Context) { |
| var query services.PipelineQuery |
| err := c.ShouldBindQuery(&query) |
| if err != nil { |
| shared.ApiOutputError(c, errors.BadInput.Wrap(err, shared.BadRequestBody)) |
| return |
| } |
| pipelines, count, err := services.GetPipelines(&query, true) |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "error getting pipelines")) |
| return |
| } |
| shared.ApiOutputSuccess(c, shared.ResponsePipelines{Pipelines: pipelines, Count: count}, http.StatusOK) |
| } |
| |
| // @Summary Get detail of a pipeline |
| // @Description GET /pipelines/:pipelineId |
| // @Description RETURN SAMPLE |
| // @Description { |
| // @Description "id": 1, |
| // @Description "name": "test-pipeline", |
| // @Description ... |
| // @Description } |
| // @Tags framework/pipelines |
| // @Param pipelineId path int true "query" |
| // @Success 200 {object} models.Pipeline |
| // @Failure 400 {string} errcode.Error "Bad Request" |
| // @Failure 500 {string} errcode.Error "Internal Error" |
| // @Router /pipelines/{pipelineId} [get] |
| func Get(c *gin.Context) { |
| pipelineId := c.Param("pipelineId") |
| id, err := strconv.ParseUint(pipelineId, 10, 64) |
| if err != nil { |
| shared.ApiOutputError(c, errors.BadInput.Wrap(err, "bad pipelineID format supplied")) |
| return |
| } |
| pipeline, err := services.GetPipeline(id, true) |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "error getting pipeline")) |
| return |
| } |
| shared.ApiOutputSuccess(c, pipeline, http.StatusOK) |
| } |
| |
| // @Summary Cancel a pending pipeline |
| // @Description Cancel a pending pipeline |
| // @Tags framework/pipelines |
| // @Param pipelineId path int true "pipeline ID" |
| // @Success 200 |
| // @Failure 400 {string} errcode.Error "Bad Request" |
| // @Failure 500 {string} errcode.Error "Internal Error" |
| // @Router /pipelines/{pipelineId} [delete] |
| func Delete(c *gin.Context) { |
| pipelineId := c.Param("pipelineId") |
| id, err := strconv.ParseUint(pipelineId, 10, 64) |
| if err != nil { |
| shared.ApiOutputError(c, errors.BadInput.Wrap(err, "bad pipelineID format supplied")) |
| return |
| } |
| err = services.CancelPipeline(id) |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "error cancelling pipeline")) |
| return |
| } |
| shared.ApiOutputSuccess(c, nil, http.StatusOK) |
| } |
| |
| // @Summary download logs of a pipeline |
| // @Description GET /pipelines/:pipelineId/logging.tar.gz |
| // @Tags framework/pipelines |
| // @Param pipelineId path int true "query" |
| // @Success 200 "The archive file" |
| // @Failure 400 {string} errcode.Error "Bad Request" |
| // @Failure 404 {string} errcode.Error "Pipeline or Log files not found" |
| // @Failure 500 {string} errcode.Error "Internal Error" |
| // @Router /pipelines/{pipelineId}/logging.tar.gz [get] |
| func DownloadLogs(c *gin.Context) { |
| pipelineId := c.Param("pipelineId") |
| id, err := strconv.ParseUint(pipelineId, 10, 64) |
| if err != nil { |
| shared.ApiOutputError(c, errors.BadInput.Wrap(err, "bad pipeline ID format supplied")) |
| return |
| } |
| pipeline, err := services.GetPipeline(id, true) |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "error getting pipeline")) |
| return |
| } |
| archive, err := services.GetPipelineLogsArchivePath(pipeline) |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "error getting logs for pipeline")) |
| return |
| } |
| defer os.Remove(archive) |
| c.FileAttachment(archive, filepath.Base(archive)) |
| } |
| |
| // RerunPipeline rerun all failed tasks of the specified pipeline |
| // @Summary rerun tasks |
| // @Tags framework/pipelines |
| // @Accept application/json |
| // @Param pipelineId path int true "pipelineId" |
| // @Success 200 {object} []models.Task |
| // @Failure 400 {object} shared.ApiBody "Bad Request" |
| // @Failure 500 {object} shared.ApiBody "Internal Error" |
| // @Router /pipelines/{pipelineId}/rerun [post] |
| func PostRerun(c *gin.Context) { |
| pipelineId := c.Param("pipelineId") |
| id, err := strconv.ParseUint(pipelineId, 10, 64) |
| if err != nil { |
| shared.ApiOutputError(c, errors.BadInput.Wrap(err, "bad pipelineID format supplied")) |
| return |
| } |
| rerunTasks, err := services.RerunPipeline(id, nil) |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "failed to rerun pipeline")) |
| return |
| } |
| for idx, task := range rerunTasks { |
| taskOption, err := services.SanitizePluginOption(task.Plugin, task.Options) |
| if err != nil { |
| shared.ApiOutputError(c, errors.Default.Wrap(err, "failed to sanitize task")) |
| return |
| } |
| task.Options = taskOption |
| rerunTasks[idx] = task |
| } |
| shared.ApiOutputSuccess(c, rerunTasks, http.StatusOK) |
| } |