mirror of https://github.com/aptly-dev/aptly
774 lines
24 KiB
Go
774 lines
24 KiB
Go
package api
|
||
|
||
import (
|
||
"fmt"
|
||
"net/http"
|
||
"os"
|
||
"path/filepath"
|
||
"sort"
|
||
"strings"
|
||
"text/template"
|
||
|
||
"github.com/aptly-dev/aptly/aptly"
|
||
"github.com/aptly-dev/aptly/database"
|
||
"github.com/aptly-dev/aptly/deb"
|
||
"github.com/aptly-dev/aptly/query"
|
||
"github.com/aptly-dev/aptly/task"
|
||
"github.com/aptly-dev/aptly/utils"
|
||
"github.com/gin-gonic/gin"
|
||
)
|
||
|
||
// GET /repos
|
||
func reposListInAPIMode(localRepos map[string]utils.FileSystemPublishRoot) gin.HandlerFunc {
|
||
return func(c *gin.Context) {
|
||
c.Writer.Header().Set("Content-Type", "text/html; charset=utf-8")
|
||
c.Writer.Flush()
|
||
c.Writer.WriteString("<pre>\n")
|
||
if len(localRepos) == 0 {
|
||
c.Writer.WriteString("<a href=\"-/\">default</a>\n")
|
||
}
|
||
for publishPrefix := range localRepos {
|
||
c.Writer.WriteString(fmt.Sprintf("<a href=\"%[1]s/\">%[1]s</a>\n", publishPrefix))
|
||
}
|
||
c.Writer.WriteString("</pre>")
|
||
c.Writer.Flush()
|
||
}
|
||
}
|
||
|
||
// GET /repos/:storage/*pkgPath
|
||
func reposServeInAPIMode(c *gin.Context) {
|
||
pkgpath := c.Param("pkgPath")
|
||
|
||
storage := c.Param("storage")
|
||
if storage == "-" {
|
||
storage = ""
|
||
} else {
|
||
storage = "filesystem:" + storage
|
||
}
|
||
|
||
publicPath := context.GetPublishedStorage(storage).(aptly.FileSystemPublishedStorage).PublicPath()
|
||
c.FileFromFS(pkgpath, http.Dir(publicPath))
|
||
}
|
||
|
||
// @Summary Get repos
|
||
// @Description Get list of available repos. Each repo is returned as in “show” API.
|
||
// @Tags Repos
|
||
// @Produce json
|
||
// @Success 200 {array} deb.LocalRepo
|
||
// @Router /api/repos [get]
|
||
func apiReposList(c *gin.Context) {
|
||
result := []*deb.LocalRepo{}
|
||
|
||
collectionFactory := context.NewCollectionFactory()
|
||
collection := collectionFactory.LocalRepoCollection()
|
||
collection.ForEach(func(r *deb.LocalRepo) error {
|
||
result = append(result, r)
|
||
return nil
|
||
})
|
||
|
||
c.JSON(200, result)
|
||
}
|
||
|
||
type repoCreateParams struct {
|
||
// Name of repository to create
|
||
Name string `binding:"required" json:"Name" example:"repo1"`
|
||
// Text describing the repository (optional)
|
||
Comment string ` json:"Comment" example:"this is a repo"`
|
||
// Default distribution when publishing from this local repo
|
||
DefaultDistribution string ` json:"DefaultDistribution" example:"stable"`
|
||
// Default component when publishing from this local repo
|
||
DefaultComponent string ` json:"DefaultComponent" example:"main"`
|
||
// Snapshot name to create repoitory from (optional)
|
||
FromSnapshot string ` json:"FromSnapshot" example:"snapshot1"`
|
||
//
|
||
LdapGroup string
|
||
}
|
||
|
||
// @Summary Create repository
|
||
// @Description Create a local repository.
|
||
// @Tags Repos
|
||
// @Produce json
|
||
// @Consume json
|
||
// @Param request body repoCreateParams true "Parameters"
|
||
// @Success 201 {object} deb.LocalRepo
|
||
// @Failure 404 {object} Error "Source snapshot not found"
|
||
// @Failure 409 {object} Error "Local repo already exists"
|
||
// @Failure 500 {object} Error "Internal error"
|
||
// @Router /api/repos [post]
|
||
func apiReposCreate(c *gin.Context) {
|
||
var b repoCreateParams
|
||
|
||
if c.Bind(&b) != nil {
|
||
return
|
||
}
|
||
|
||
repo := deb.NewLocalRepo(b.Name, b.Comment)
|
||
repo.DefaultComponent = b.DefaultComponent
|
||
repo.DefaultDistribution = b.DefaultDistribution
|
||
repo.LdapGroup = b.LdapGroup
|
||
|
||
collectionFactory := context.NewCollectionFactory()
|
||
|
||
if b.FromSnapshot != "" {
|
||
var snapshot *deb.Snapshot
|
||
|
||
snapshotCollection := collectionFactory.SnapshotCollection()
|
||
|
||
snapshot, err := snapshotCollection.ByName(b.FromSnapshot)
|
||
if err != nil {
|
||
AbortWithJSONError(c, http.StatusNotFound, fmt.Errorf("source snapshot not found: %s", err))
|
||
return
|
||
}
|
||
|
||
err = snapshotCollection.LoadComplete(snapshot)
|
||
if err != nil {
|
||
AbortWithJSONError(c, http.StatusInternalServerError, fmt.Errorf("unable to load source snapshot: %s", err))
|
||
return
|
||
}
|
||
|
||
repo.UpdateRefList(snapshot.RefList())
|
||
}
|
||
|
||
localRepoCollection := collectionFactory.LocalRepoCollection()
|
||
|
||
if _, err := localRepoCollection.ByName(b.Name); err == nil {
|
||
AbortWithJSONError(c, http.StatusConflict, fmt.Errorf("local repo with name %s already exists", b.Name))
|
||
return
|
||
}
|
||
|
||
err := localRepoCollection.Add(repo)
|
||
if err != nil {
|
||
AbortWithJSONError(c, http.StatusInternalServerError, err)
|
||
return
|
||
}
|
||
|
||
c.JSON(http.StatusCreated, repo)
|
||
}
|
||
|
||
// PUT /api/repos/:name
|
||
func apiReposEdit(c *gin.Context) {
|
||
var b struct {
|
||
Name *string
|
||
Comment *string
|
||
DefaultDistribution *string
|
||
DefaultComponent *string
|
||
LdapGroup *string
|
||
}
|
||
|
||
if c.Bind(&b) != nil {
|
||
return
|
||
}
|
||
|
||
collectionFactory := context.NewCollectionFactory()
|
||
collection := collectionFactory.LocalRepoCollection()
|
||
|
||
repo, err := collection.ByName(c.Params.ByName("name"))
|
||
if err != nil {
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
|
||
err = CheckGroup(c, repo.LdapGroup)
|
||
if err != nil {
|
||
c.AbortWithError(403, err)
|
||
return
|
||
}
|
||
|
||
if b.Name != nil {
|
||
_, err := collection.ByName(*b.Name)
|
||
if err == nil {
|
||
// already exists
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
repo.Name = *b.Name
|
||
}
|
||
if b.Comment != nil {
|
||
repo.Comment = *b.Comment
|
||
}
|
||
if b.DefaultDistribution != nil {
|
||
repo.DefaultDistribution = *b.DefaultDistribution
|
||
}
|
||
if b.DefaultComponent != nil {
|
||
repo.DefaultComponent = *b.DefaultComponent
|
||
}
|
||
if b.LdapGroup != nil {
|
||
repo.LdapGroup = *b.LdapGroup
|
||
}
|
||
|
||
err = collection.Update(repo)
|
||
if err != nil {
|
||
AbortWithJSONError(c, 500, err)
|
||
return
|
||
}
|
||
|
||
c.JSON(200, repo)
|
||
}
|
||
|
||
// GET /api/repos/:name
|
||
// @Summary Get repository info by name
|
||
// @Description Returns basic information about local repository.
|
||
// @Tags Repos
|
||
// @Produce json
|
||
// @Param name path string true "Repository name"
|
||
// @Success 200 {object} deb.LocalRepo
|
||
// @Failure 404 {object} Error "Repository not found"
|
||
// @Router /api/repos/{name} [get]
|
||
func apiReposShow(c *gin.Context) {
|
||
collectionFactory := context.NewCollectionFactory()
|
||
collection := collectionFactory.LocalRepoCollection()
|
||
|
||
repo, err := collection.ByName(c.Params.ByName("name"))
|
||
if err != nil {
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
|
||
c.JSON(200, repo)
|
||
}
|
||
|
||
// DELETE /api/repos/:name
|
||
func apiReposDrop(c *gin.Context) {
|
||
force := c.Request.URL.Query().Get("force") == "1"
|
||
name := c.Params.ByName("name")
|
||
|
||
collectionFactory := context.NewCollectionFactory()
|
||
collection := collectionFactory.LocalRepoCollection()
|
||
snapshotCollection := collectionFactory.SnapshotCollection()
|
||
publishedCollection := collectionFactory.PublishedRepoCollection()
|
||
|
||
repo, err := collection.ByName(name)
|
||
if err != nil {
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
|
||
err = CheckGroup(c, repo.LdapGroup)
|
||
if err != nil {
|
||
c.AbortWithError(403, err)
|
||
return
|
||
}
|
||
|
||
resources := []string{string(repo.Key())}
|
||
taskName := fmt.Sprintf("Delete repo %s", name)
|
||
maybeRunTaskInBackground(c, taskName, resources, func(_ aptly.Progress, _ *task.Detail) (*task.ProcessReturnValue, error) {
|
||
published := publishedCollection.ByLocalRepo(repo)
|
||
if len(published) > 0 {
|
||
return &task.ProcessReturnValue{Code: http.StatusConflict, Value: nil}, fmt.Errorf("unable to drop, local repo is published")
|
||
}
|
||
|
||
if !force {
|
||
snapshots := snapshotCollection.ByLocalRepoSource(repo)
|
||
if len(snapshots) > 0 {
|
||
return &task.ProcessReturnValue{Code: http.StatusConflict, Value: nil}, fmt.Errorf("unable to drop, local repo has snapshots, use ?force=1 to override")
|
||
}
|
||
}
|
||
|
||
return &task.ProcessReturnValue{Code: http.StatusOK, Value: gin.H{}}, collection.Drop(repo)
|
||
})
|
||
}
|
||
|
||
// GET /api/repos/:name/packages
|
||
func apiReposPackagesShow(c *gin.Context) {
|
||
collectionFactory := context.NewCollectionFactory()
|
||
collection := collectionFactory.LocalRepoCollection()
|
||
|
||
repo, err := collection.ByName(c.Params.ByName("name"))
|
||
if err != nil {
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
|
||
err = collection.LoadComplete(repo)
|
||
if err != nil {
|
||
AbortWithJSONError(c, 500, err)
|
||
return
|
||
}
|
||
|
||
showPackages(c, repo.RefList(), collectionFactory)
|
||
}
|
||
|
||
// Handler for both add and delete
|
||
func apiReposPackagesAddDelete(c *gin.Context, taskNamePrefix string, cb func(list *deb.PackageList, p *deb.Package, out aptly.Progress) error) {
|
||
var b struct {
|
||
PackageRefs []string
|
||
}
|
||
|
||
if c.Bind(&b) != nil {
|
||
return
|
||
}
|
||
|
||
collectionFactory := context.NewCollectionFactory()
|
||
collection := collectionFactory.LocalRepoCollection()
|
||
|
||
repo, err := collection.ByName(c.Params.ByName("name"))
|
||
if err != nil {
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
|
||
resources := []string{string(repo.Key())}
|
||
|
||
maybeRunTaskInBackground(c, taskNamePrefix+repo.Name, resources, func(out aptly.Progress, _ *task.Detail) (*task.ProcessReturnValue, error) {
|
||
err = collection.LoadComplete(repo)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, err
|
||
}
|
||
|
||
err = CheckGroup(c, repo.LdapGroup)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: 403, Value: nil}, err
|
||
}
|
||
|
||
out.Printf("Loading packages...\n")
|
||
list, err := deb.NewPackageListFromRefList(repo.RefList(), collectionFactory.PackageCollection(), nil)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, err
|
||
}
|
||
|
||
// verify package refs and build package list
|
||
for _, ref := range b.PackageRefs {
|
||
var p *deb.Package
|
||
|
||
p, err = collectionFactory.PackageCollection().ByKey([]byte(ref))
|
||
if err != nil {
|
||
if err == database.ErrNotFound {
|
||
return &task.ProcessReturnValue{Code: http.StatusNotFound, Value: nil}, fmt.Errorf("packages %s: %s", ref, err)
|
||
}
|
||
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, err
|
||
}
|
||
err = cb(list, p, out)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusBadRequest, Value: nil}, err
|
||
}
|
||
}
|
||
|
||
repo.UpdateRefList(deb.NewPackageRefListFromPackageList(list))
|
||
|
||
err = collectionFactory.LocalRepoCollection().Update(repo)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to save: %s", err)
|
||
}
|
||
return &task.ProcessReturnValue{Code: http.StatusOK, Value: repo}, nil
|
||
})
|
||
}
|
||
|
||
// POST /repos/:name/packages
|
||
func apiReposPackagesAdd(c *gin.Context) {
|
||
apiReposPackagesAddDelete(c, "Add packages to repo ", func(list *deb.PackageList, p *deb.Package, out aptly.Progress) error {
|
||
out.Printf("Adding package %s\n", p.Name)
|
||
return list.Add(p)
|
||
})
|
||
}
|
||
|
||
// DELETE /repos/:name/packages
|
||
func apiReposPackagesDelete(c *gin.Context) {
|
||
apiReposPackagesAddDelete(c, "Delete packages from repo ", func(list *deb.PackageList, p *deb.Package, out aptly.Progress) error {
|
||
out.Printf("Removing package %s\n", p.Name)
|
||
list.Remove(p)
|
||
return nil
|
||
})
|
||
}
|
||
|
||
// POST /repos/:name/file/:dir/:file
|
||
func apiReposPackageFromFile(c *gin.Context) {
|
||
// redirect all work to dir method
|
||
apiReposPackageFromDir(c)
|
||
}
|
||
|
||
// @Summary Add packages from uploaded file/directory
|
||
// @Description Import packages from files (uploaded using File Upload API) to the local repository. If directory specified, aptly would discover package files automatically.
|
||
// @Description Adding same package to local repository is not an error.
|
||
// @Description By default aptly would try to remove every successfully processed file and directory `dir` (if it becomes empty after import).
|
||
// @Tags Repos
|
||
// @Param name path string true "Repository name"
|
||
// @Param dir path string true "Directory to add"
|
||
// @Consume json
|
||
// @Param noRemove query string false "when value is set to 1, don’t remove any files"
|
||
// @Param forceReplace query string false "when value is set to 1, remove packages conflicting with package being added (in local repository)"
|
||
// @Produce json
|
||
// @Success 200 {string} string "OK"
|
||
// @Failure 400 {object} Error "wrong file"
|
||
// @Failure 404 {object} Error "Repository not found"
|
||
// @Failure 500 {object} Error "Error adding files"
|
||
// @Router /api/repos/{name}/{dir} [post]
|
||
func apiReposPackageFromDir(c *gin.Context) {
|
||
forceReplace := c.Request.URL.Query().Get("forceReplace") == "1"
|
||
noRemove := c.Request.URL.Query().Get("noRemove") == "1"
|
||
|
||
if !verifyDir(c) {
|
||
return
|
||
}
|
||
|
||
dirParam := utils.SanitizePath(c.Params.ByName("dir"))
|
||
fileParam := utils.SanitizePath(c.Params.ByName("file"))
|
||
if fileParam != "" && !verifyPath(fileParam) {
|
||
AbortWithJSONError(c, 400, fmt.Errorf("wrong file"))
|
||
return
|
||
}
|
||
|
||
collectionFactory := context.NewCollectionFactory()
|
||
collection := collectionFactory.LocalRepoCollection()
|
||
|
||
name := c.Params.ByName("name")
|
||
repo, err := collection.ByName(name)
|
||
if err != nil {
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
|
||
var taskName string
|
||
var sources []string
|
||
if fileParam == "" {
|
||
taskName = fmt.Sprintf("Add packages from dir %s to repo %s", dirParam, name)
|
||
sources = []string{filepath.Join(context.UploadPath(), dirParam)}
|
||
} else {
|
||
sources = []string{filepath.Join(context.UploadPath(), dirParam, fileParam)}
|
||
taskName = fmt.Sprintf("Add package %s from dir %s to repo %s", fileParam, dirParam, name)
|
||
}
|
||
|
||
resources := []string{string(repo.Key())}
|
||
resources = append(resources, sources...)
|
||
maybeRunTaskInBackground(c, taskName, resources, func(out aptly.Progress, _ *task.Detail) (*task.ProcessReturnValue, error) {
|
||
err = collection.LoadComplete(repo)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, err
|
||
}
|
||
|
||
err = CheckGroup(c, repo.LdapGroup)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: 403, Value: nil}, err
|
||
}
|
||
|
||
verifier := context.GetVerifier()
|
||
|
||
var (
|
||
packageFiles, failedFiles []string
|
||
otherFiles []string
|
||
processedFiles, failedFiles2 []string
|
||
reporter = &aptly.RecordingResultReporter{
|
||
Warnings: []string{},
|
||
AddedLines: []string{},
|
||
RemovedLines: []string{},
|
||
}
|
||
list *deb.PackageList
|
||
)
|
||
|
||
packageFiles, otherFiles, failedFiles = deb.CollectPackageFiles(sources, reporter)
|
||
|
||
list, err := deb.NewPackageListFromRefList(repo.RefList(), collectionFactory.PackageCollection(), nil)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to load packages: %s", err)
|
||
}
|
||
|
||
processedFiles, failedFiles2, err = deb.ImportPackageFiles(list, packageFiles, forceReplace, verifier, context.PackagePool(),
|
||
collectionFactory.PackageCollection(), reporter, nil, collectionFactory.ChecksumCollection)
|
||
failedFiles = append(failedFiles, failedFiles2...)
|
||
processedFiles = append(processedFiles, otherFiles...)
|
||
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to import package files: %s", err)
|
||
}
|
||
|
||
repo.UpdateRefList(deb.NewPackageRefListFromPackageList(list))
|
||
|
||
err = collectionFactory.LocalRepoCollection().Update(repo)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to save: %s", err)
|
||
}
|
||
|
||
if !noRemove {
|
||
processedFiles = utils.StrSliceDeduplicate(processedFiles)
|
||
|
||
for _, file := range processedFiles {
|
||
err := os.Remove(file)
|
||
if err != nil {
|
||
reporter.Warning("unable to remove file %s: %s", file, err)
|
||
}
|
||
}
|
||
|
||
// atempt to remove dir, if it fails, that's fine: probably it's not empty
|
||
os.Remove(filepath.Join(context.UploadPath(), dirParam))
|
||
}
|
||
|
||
if failedFiles == nil {
|
||
failedFiles = []string{}
|
||
}
|
||
|
||
if len(reporter.AddedLines) > 0 {
|
||
out.Printf("Added: %s\n", strings.Join(reporter.AddedLines, ", "))
|
||
}
|
||
if len(reporter.RemovedLines) > 0 {
|
||
out.Printf("Removed: %s\n", strings.Join(reporter.RemovedLines, ", "))
|
||
}
|
||
if len(reporter.Warnings) > 0 {
|
||
out.Printf("Warnings: %s\n", strings.Join(reporter.Warnings, ", "))
|
||
}
|
||
if len(failedFiles) > 0 {
|
||
out.Printf("Failed files: %s\n", strings.Join(failedFiles, ", "))
|
||
}
|
||
|
||
return &task.ProcessReturnValue{Code: http.StatusOK, Value: gin.H{
|
||
"Report": reporter,
|
||
"FailedFiles": failedFiles,
|
||
}}, nil
|
||
})
|
||
}
|
||
|
||
// POST /repos/:name/copy/:src/:file
|
||
func apiReposCopyPackage(c *gin.Context) {
|
||
dstRepoName := c.Params.ByName("name")
|
||
srcRepoName := c.Params.ByName("src")
|
||
fileName := c.Params.ByName("file")
|
||
|
||
jsonBody := struct {
|
||
WithDeps bool `json:"with-deps,omitempty"`
|
||
DryRun bool `json:"dry-run,omitempty"`
|
||
}{
|
||
WithDeps: false,
|
||
DryRun: false,
|
||
}
|
||
|
||
err := c.Bind(&jsonBody)
|
||
if err != nil {
|
||
return
|
||
}
|
||
|
||
collectionFactory := context.NewCollectionFactory()
|
||
dstRepo, err := collectionFactory.LocalRepoCollection().ByName(dstRepoName)
|
||
if err != nil {
|
||
AbortWithJSONError(c, http.StatusBadRequest, fmt.Errorf("dest repo error: %s", err))
|
||
return
|
||
}
|
||
|
||
var srcRepo *deb.LocalRepo
|
||
srcRepo, err = collectionFactory.LocalRepoCollection().ByName(srcRepoName)
|
||
if err != nil {
|
||
AbortWithJSONError(c, http.StatusBadRequest, fmt.Errorf("src repo error: %s", err))
|
||
return
|
||
}
|
||
|
||
if srcRepo.UUID == dstRepo.UUID {
|
||
AbortWithJSONError(c, http.StatusBadRequest, fmt.Errorf("dest and source are identical"))
|
||
return
|
||
}
|
||
|
||
taskName := fmt.Sprintf("Copy packages from repo %s to repo %s", srcRepoName, dstRepoName)
|
||
resources := []string{string(dstRepo.Key()), string(srcRepo.Key())}
|
||
|
||
maybeRunTaskInBackground(c, taskName, resources, func(_ aptly.Progress, _ *task.Detail) (*task.ProcessReturnValue, error) {
|
||
err = collectionFactory.LocalRepoCollection().LoadComplete(dstRepo)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusBadRequest, Value: nil}, fmt.Errorf("dest repo error: %s", err)
|
||
}
|
||
|
||
err = collectionFactory.LocalRepoCollection().LoadComplete(srcRepo)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusBadRequest, Value: nil}, fmt.Errorf("src repo error: %s", err)
|
||
}
|
||
|
||
srcRefList := srcRepo.RefList()
|
||
|
||
reporter := &aptly.RecordingResultReporter{
|
||
Warnings: []string{},
|
||
AddedLines: []string{},
|
||
RemovedLines: []string{},
|
||
}
|
||
|
||
dstList, err := deb.NewPackageListFromRefList(dstRepo.RefList(), collectionFactory.PackageCollection(), context.Progress())
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to load packages in dest: %s", err)
|
||
|
||
}
|
||
|
||
srcList, err := deb.NewPackageListFromRefList(srcRefList, collectionFactory.PackageCollection(), context.Progress())
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to load packages in src: %s", err)
|
||
}
|
||
|
||
srcList.PrepareIndex()
|
||
|
||
var architecturesList []string
|
||
|
||
if jsonBody.WithDeps {
|
||
dstList.PrepareIndex()
|
||
|
||
// Calculate architectures
|
||
if len(context.ArchitecturesList()) > 0 {
|
||
architecturesList = context.ArchitecturesList()
|
||
} else {
|
||
architecturesList = dstList.Architectures(false)
|
||
}
|
||
|
||
sort.Strings(architecturesList)
|
||
|
||
if len(architecturesList) == 0 {
|
||
return &task.ProcessReturnValue{Code: http.StatusUnprocessableEntity, Value: nil}, fmt.Errorf("unable to determine list of architectures, please specify explicitly")
|
||
}
|
||
}
|
||
|
||
// srcList.Filter|FilterWithProgress only accept query list
|
||
queries := make([]deb.PackageQuery, 1)
|
||
queries[0], err = query.Parse(fileName)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusUnprocessableEntity, Value: nil}, fmt.Errorf("unable to parse query '%s': %s", fileName, err)
|
||
}
|
||
|
||
toProcess, err := srcList.FilterWithProgress(queries, jsonBody.WithDeps, dstList, context.DependencyOptions(), architecturesList, context.Progress())
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("filter error: %s", err)
|
||
}
|
||
|
||
if toProcess.Len() == 0 {
|
||
return &task.ProcessReturnValue{Code: http.StatusUnprocessableEntity, Value: nil}, fmt.Errorf("no package found for filter: '%s'", fileName)
|
||
}
|
||
|
||
err = toProcess.ForEach(func(p *deb.Package) error {
|
||
err = dstList.Add(p)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
name := fmt.Sprintf("added %s-%s(%s)", p.Name, p.Version, p.Architecture)
|
||
reporter.AddedLines = append(reporter.AddedLines, name)
|
||
return nil
|
||
})
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("error processing dest add: %s", err)
|
||
}
|
||
|
||
if jsonBody.DryRun {
|
||
reporter.Warning("Changes not saved, as dry run has been requested")
|
||
} else {
|
||
dstRepo.UpdateRefList(deb.NewPackageRefListFromPackageList(dstList))
|
||
|
||
err = collectionFactory.LocalRepoCollection().Update(dstRepo)
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to save: %s", err)
|
||
}
|
||
}
|
||
|
||
return &task.ProcessReturnValue{Code: http.StatusOK, Value: gin.H{
|
||
"Report": reporter,
|
||
}}, nil
|
||
})
|
||
}
|
||
|
||
// POST /repos/:name/include/:dir/:file
|
||
func apiReposIncludePackageFromFile(c *gin.Context) {
|
||
// redirect all work to dir method
|
||
apiReposIncludePackageFromDir(c)
|
||
}
|
||
|
||
// POST /repos/:name/include/:dir
|
||
func apiReposIncludePackageFromDir(c *gin.Context) {
|
||
forceReplace := c.Request.URL.Query().Get("forceReplace") == "1"
|
||
noRemoveFiles := c.Request.URL.Query().Get("noRemoveFiles") == "1"
|
||
acceptUnsigned := c.Request.URL.Query().Get("acceptUnsigned") == "1"
|
||
ignoreSignature := c.Request.URL.Query().Get("ignoreSignature") == "1"
|
||
|
||
repoTemplateString := c.Params.ByName("name")
|
||
collectionFactory := context.NewCollectionFactory()
|
||
|
||
if !verifyDir(c) {
|
||
return
|
||
}
|
||
|
||
var sources []string
|
||
var taskName string
|
||
dirParam := utils.SanitizePath(c.Params.ByName("dir"))
|
||
fileParam := utils.SanitizePath(c.Params.ByName("file"))
|
||
if fileParam != "" && !verifyPath(fileParam) {
|
||
AbortWithJSONError(c, 400, fmt.Errorf("wrong file"))
|
||
return
|
||
}
|
||
|
||
if fileParam == "" {
|
||
taskName = fmt.Sprintf("Include packages from changes files in dir %s to repo matching template %s", dirParam, repoTemplateString)
|
||
sources = []string{filepath.Join(context.UploadPath(), dirParam)}
|
||
} else {
|
||
taskName = fmt.Sprintf("Include packages from changes file %s from dir %s to repo matching template %s", fileParam, dirParam, repoTemplateString)
|
||
sources = []string{filepath.Join(context.UploadPath(), dirParam, fileParam)}
|
||
}
|
||
|
||
repoTemplate, err := template.New("repo").Parse(repoTemplateString)
|
||
if err != nil {
|
||
AbortWithJSONError(c, 400, fmt.Errorf("error parsing repo template: %s", err))
|
||
return
|
||
}
|
||
|
||
var resources []string
|
||
if len(repoTemplate.Tree.Root.Nodes) > 1 {
|
||
resources = append(resources, task.AllLocalReposResourcesKey)
|
||
} else {
|
||
// repo template string is simple text so only use resource key of specific repository
|
||
repo, err := collectionFactory.LocalRepoCollection().ByName(repoTemplateString)
|
||
if err != nil {
|
||
AbortWithJSONError(c, 404, err)
|
||
return
|
||
}
|
||
err = CheckGroup(c, repo.LdapGroup)
|
||
if err != nil {
|
||
c.AbortWithError(403, err)
|
||
return
|
||
}
|
||
|
||
resources = append(resources, string(repo.Key()))
|
||
}
|
||
resources = append(resources, sources...)
|
||
|
||
maybeRunTaskInBackground(c, taskName, resources, func(out aptly.Progress, _ *task.Detail) (*task.ProcessReturnValue, error) {
|
||
var (
|
||
err error
|
||
verifier = context.GetVerifier()
|
||
changesFiles []string
|
||
failedFiles, failedFiles2 []string
|
||
reporter = &aptly.RecordingResultReporter{
|
||
Warnings: []string{},
|
||
AddedLines: []string{},
|
||
RemovedLines: []string{},
|
||
}
|
||
)
|
||
|
||
changesFiles, failedFiles = deb.CollectChangesFiles(sources, reporter)
|
||
_, failedFiles2, err = deb.ImportChangesFiles(
|
||
changesFiles, reporter, acceptUnsigned, ignoreSignature, forceReplace, noRemoveFiles, verifier,
|
||
repoTemplate, context.Progress(), collectionFactory.LocalRepoCollection(), collectionFactory.PackageCollection(),
|
||
context.PackagePool(), collectionFactory.ChecksumCollection, nil, query.Parse)
|
||
failedFiles = append(failedFiles, failedFiles2...)
|
||
|
||
if err != nil {
|
||
return &task.ProcessReturnValue{Code: http.StatusInternalServerError, Value: nil}, fmt.Errorf("unable to import changes files: %s", err)
|
||
}
|
||
|
||
if !noRemoveFiles {
|
||
// atempt to remove dir, if it fails, that's fine: probably it's not empty
|
||
os.Remove(filepath.Join(context.UploadPath(), dirParam))
|
||
}
|
||
|
||
if failedFiles == nil {
|
||
failedFiles = []string{}
|
||
}
|
||
|
||
if len(reporter.AddedLines) > 0 {
|
||
out.Printf("Added: %s\n", strings.Join(reporter.AddedLines, ", "))
|
||
}
|
||
if len(reporter.RemovedLines) > 0 {
|
||
out.Printf("Removed: %s\n", strings.Join(reporter.RemovedLines, ", "))
|
||
}
|
||
if len(reporter.Warnings) > 0 {
|
||
out.Printf("Warnings: %s\n", strings.Join(reporter.Warnings, ", "))
|
||
}
|
||
if len(failedFiles) > 0 {
|
||
out.Printf("Failed files: %s\n", strings.Join(failedFiles, ", "))
|
||
}
|
||
|
||
return &task.ProcessReturnValue{Code: http.StatusOK, Value: gin.H{
|
||
"Report": reporter,
|
||
"FailedFiles": failedFiles,
|
||
}}, nil
|
||
|
||
})
|
||
}
|