mirror of https://github.com/aptly-dev/aptly
376 lines
11 KiB
Go
376 lines
11 KiB
Go
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
|
|
"github.com/aptly-dev/aptly/aptly"
|
|
"github.com/aptly-dev/aptly/database"
|
|
"github.com/aptly-dev/aptly/deb"
|
|
"github.com/aptly-dev/aptly/utils"
|
|
"github.com/smira/commander"
|
|
)
|
|
|
|
// aptly db cleanup
|
|
func aptlyDbCleanup(cmd *commander.Command, args []string) error {
|
|
var err error
|
|
|
|
if len(args) != 0 {
|
|
cmd.Usage()
|
|
return commander.ErrCommandError
|
|
}
|
|
|
|
verbose := context.Flags().Lookup("verbose").Value.Get().(bool)
|
|
dryRun := context.Flags().Lookup("dry-run").Value.Get().(bool)
|
|
collectionFactory := context.NewCollectionFactory()
|
|
|
|
// collect information about references packages and their reflistbuckets...
|
|
existingPackageRefs := deb.NewSplitRefList()
|
|
existingBuckets := deb.NewRefListDigestSet()
|
|
|
|
// used only in verbose mode to report package use source
|
|
packageRefSources := map[string][]string{}
|
|
|
|
var reflistMigration *deb.RefListMigration
|
|
if !dryRun {
|
|
reflistMigration = collectionFactory.RefListCollection().NewMigration()
|
|
} else {
|
|
reflistMigration = collectionFactory.RefListCollection().NewMigrationDryRun()
|
|
}
|
|
|
|
context.Progress().ColoredPrintf("@{w!}Loading mirrors, local repos, snapshots and published repos...@|")
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("@{y}Loading mirrors:@|")
|
|
}
|
|
err = collectionFactory.RemoteRepoCollection().ForEach(func(repo *deb.RemoteRepo) error {
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("- @{g}%s@|", repo.Name)
|
|
}
|
|
|
|
sl := deb.NewSplitRefList()
|
|
e := collectionFactory.RefListCollection().LoadCompleteAndMigrate(sl, repo.RefKey(), reflistMigration)
|
|
if e != nil && e != database.ErrNotFound {
|
|
return e
|
|
}
|
|
|
|
existingPackageRefs = existingPackageRefs.Merge(sl, false, true)
|
|
existingBuckets.AddAllInRefList(sl)
|
|
|
|
if verbose {
|
|
description := fmt.Sprintf("mirror %s", repo.Name)
|
|
sl.ForEach(func(key []byte) error {
|
|
packageRefSources[string(key)] = append(packageRefSources[string(key)], description)
|
|
return nil
|
|
})
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
collectionFactory.Flush()
|
|
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("@{y}Loading local repos:@|")
|
|
}
|
|
err = collectionFactory.LocalRepoCollection().ForEach(func(repo *deb.LocalRepo) error {
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("- @{g}%s@|", repo.Name)
|
|
}
|
|
|
|
sl := deb.NewSplitRefList()
|
|
e := collectionFactory.RefListCollection().LoadCompleteAndMigrate(sl, repo.RefKey(), reflistMigration)
|
|
if e != nil && e != database.ErrNotFound {
|
|
return e
|
|
}
|
|
|
|
existingPackageRefs = existingPackageRefs.Merge(sl, false, true)
|
|
existingBuckets.AddAllInRefList(sl)
|
|
|
|
existingPackageRefs = existingPackageRefs.Merge(sl, false, true)
|
|
|
|
if verbose {
|
|
description := fmt.Sprintf("local repo %s", repo.Name)
|
|
sl.ForEach(func(key []byte) error {
|
|
packageRefSources[string(key)] = append(packageRefSources[string(key)], description)
|
|
return nil
|
|
})
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
collectionFactory.Flush()
|
|
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("@{y}Loading snapshots:@|")
|
|
}
|
|
err = collectionFactory.SnapshotCollection().ForEach(func(snapshot *deb.Snapshot) error {
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("- @{g}%s@|", snapshot.Name)
|
|
}
|
|
|
|
sl := deb.NewSplitRefList()
|
|
e := collectionFactory.RefListCollection().LoadCompleteAndMigrate(sl, snapshot.RefKey(), reflistMigration)
|
|
if e != nil {
|
|
return e
|
|
}
|
|
|
|
existingPackageRefs = existingPackageRefs.Merge(sl, false, true)
|
|
existingBuckets.AddAllInRefList(sl)
|
|
|
|
if verbose {
|
|
description := fmt.Sprintf("snapshot %s", snapshot.Name)
|
|
sl.ForEach(func(key []byte) error {
|
|
packageRefSources[string(key)] = append(packageRefSources[string(key)], description)
|
|
return nil
|
|
})
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
collectionFactory.Flush()
|
|
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("@{y}Loading published repositories:@|")
|
|
}
|
|
err = collectionFactory.PublishedRepoCollection().ForEach(func(published *deb.PublishedRepo) error {
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("- @{g}%s:%s/%s{|}", published.Storage, published.Prefix, published.Distribution)
|
|
}
|
|
if published.SourceKind != deb.SourceLocalRepo {
|
|
return nil
|
|
}
|
|
|
|
for _, component := range published.Components() {
|
|
sl := deb.NewSplitRefList()
|
|
e := collectionFactory.RefListCollection().LoadCompleteAndMigrate(sl, published.RefKey(component), reflistMigration)
|
|
if e != nil {
|
|
return e
|
|
}
|
|
|
|
existingPackageRefs = existingPackageRefs.Merge(sl, false, true)
|
|
existingBuckets.AddAllInRefList(sl)
|
|
|
|
if verbose {
|
|
description := fmt.Sprintf("published repository %s:%s/%s component %s",
|
|
published.Storage, published.Prefix, published.Distribution, component)
|
|
sl.ForEach(func(key []byte) error {
|
|
packageRefSources[string(key)] = append(packageRefSources[string(key)], description)
|
|
return nil
|
|
})
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
collectionFactory.Flush()
|
|
|
|
err = reflistMigration.Flush()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if verbose {
|
|
if stats := reflistMigration.Stats(); stats.Reflists > 0 {
|
|
if !dryRun {
|
|
context.Progress().ColoredPrintf("@{w!}Split %d reflist(s) into %d bucket(s) (%d segment(s))@|",
|
|
stats.Reflists, stats.Buckets, stats.Segments)
|
|
} else {
|
|
context.Progress().ColoredPrintf(
|
|
"@{y!}Skipped splitting %d reflist(s) into %d bucket(s) (%d segment(s)), as -dry-run has been requested.@|",
|
|
stats.Reflists, stats.Buckets, stats.Segments)
|
|
}
|
|
}
|
|
}
|
|
|
|
// ... and compare it to the list of all packages
|
|
context.Progress().ColoredPrintf("@{w!}Loading list of all packages...@|")
|
|
allPackageRefs := collectionFactory.PackageCollection().AllPackageRefs()
|
|
|
|
toDelete := allPackageRefs.Subtract(existingPackageRefs.Flatten())
|
|
|
|
// delete packages that are no longer referenced
|
|
context.Progress().ColoredPrintf("@{r!}Deleting unreferenced packages (%d)...@|", toDelete.Len())
|
|
|
|
// database can't err as collection factory already constructed
|
|
db, _ := context.Database()
|
|
|
|
if toDelete.Len() > 0 {
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("@{r}List of package keys to delete:@|")
|
|
err = toDelete.ForEach(func(ref []byte) error {
|
|
context.Progress().ColoredPrintf(" - @{r}%s@|", string(ref))
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if !dryRun {
|
|
batch := db.CreateBatch()
|
|
err = toDelete.ForEach(func(ref []byte) error {
|
|
return collectionFactory.PackageCollection().DeleteByKey(ref, batch)
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("unable to delete by key: %s", err)
|
|
}
|
|
|
|
err = batch.Write()
|
|
if err != nil {
|
|
return fmt.Errorf("unable to write to DB: %s", err)
|
|
}
|
|
} else {
|
|
context.Progress().ColoredPrintf("@{y!}Skipped deletion, as -dry-run has been requested.@|")
|
|
}
|
|
}
|
|
|
|
bucketsToDelete, err := collectionFactory.RefListCollection().AllBucketDigests()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
bucketsToDelete.RemoveAll(existingBuckets)
|
|
|
|
context.Progress().ColoredPrintf("@{r!}Deleting unreferenced reflist buckets (%d)...@|", bucketsToDelete.Len())
|
|
if bucketsToDelete.Len() > 0 {
|
|
if !dryRun {
|
|
batch := db.CreateBatch()
|
|
err := bucketsToDelete.ForEach(func(digest []byte) error {
|
|
return collectionFactory.RefListCollection().UnsafeDropBucket(digest, batch)
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := batch.Write(); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
context.Progress().ColoredPrintf("@{y!}Skipped reflist deletion, as -dry-run has been requested.@|")
|
|
}
|
|
}
|
|
|
|
collectionFactory.Flush()
|
|
|
|
// now, build a list of files that should be present in Repository (package pool)
|
|
context.Progress().ColoredPrintf("@{w!}Building list of files referenced by packages...@|")
|
|
referencedFiles := make([]string, 0, existingPackageRefs.Len())
|
|
context.Progress().InitBar(int64(existingPackageRefs.Len()), false, aptly.BarCleanupBuildList)
|
|
|
|
err = existingPackageRefs.ForEach(func(key []byte) error {
|
|
pkg, err2 := collectionFactory.PackageCollection().ByKey(key)
|
|
if err2 != nil {
|
|
tail := ""
|
|
if verbose {
|
|
tail = fmt.Sprintf(" (sources: %s)", strings.Join(packageRefSources[string(key)], ", "))
|
|
}
|
|
if dryRun {
|
|
context.Progress().ColoredPrintf("@{r!}Unresolvable package reference, skipping (-dry-run): %s: %s%s",
|
|
string(key), err2, tail)
|
|
return nil
|
|
}
|
|
return fmt.Errorf("unable to load package %s: %s%s", string(key), err2, tail)
|
|
}
|
|
paths, err2 := pkg.FilepathList(context.PackagePool())
|
|
if err2 != nil {
|
|
return err2
|
|
}
|
|
referencedFiles = append(referencedFiles, paths...)
|
|
context.Progress().AddBar(1)
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
sort.Strings(referencedFiles)
|
|
context.Progress().ShutdownBar()
|
|
|
|
// build a list of files in the package pool
|
|
context.Progress().ColoredPrintf("@{w!}Building list of files in package pool...@|")
|
|
existingFiles, err := context.PackagePool().FilepathList(context.Progress())
|
|
if err != nil {
|
|
return fmt.Errorf("unable to collect file paths: %s", err)
|
|
}
|
|
|
|
// find files which are in the pool but not referenced by packages
|
|
filesToDelete := utils.StrSlicesSubstract(existingFiles, referencedFiles)
|
|
|
|
// delete files that are no longer referenced
|
|
context.Progress().ColoredPrintf("@{r!}Deleting unreferenced files (%d)...@|", len(filesToDelete))
|
|
|
|
if len(filesToDelete) > 0 {
|
|
if verbose {
|
|
context.Progress().ColoredPrintf("@{r}List of files to be deleted:@|")
|
|
for _, file := range filesToDelete {
|
|
context.Progress().ColoredPrintf(" - @{r}%s@|", file)
|
|
}
|
|
}
|
|
|
|
if !dryRun {
|
|
context.Progress().InitBar(int64(len(filesToDelete)), false, aptly.BarCleanupDeleteUnreferencedFiles)
|
|
|
|
var size, totalSize int64
|
|
for _, file := range filesToDelete {
|
|
size, err = context.PackagePool().Remove(file)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
context.Progress().AddBar(1)
|
|
totalSize += size
|
|
}
|
|
context.Progress().ShutdownBar()
|
|
|
|
context.Progress().ColoredPrintf("@{w!}Disk space freed: %s...@|", utils.HumanBytes(totalSize))
|
|
} else {
|
|
context.Progress().ColoredPrintf("@{y!}Skipped file deletion, as -dry-run has been requested.@|")
|
|
}
|
|
}
|
|
|
|
if !dryRun {
|
|
context.Progress().ColoredPrintf("@{w!}Compacting database...@|")
|
|
err = db.CompactDB()
|
|
} else {
|
|
context.Progress().ColoredPrintf("@{y!}Skipped DB compaction, as -dry-run has been requested.@|")
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
func makeCmdDbCleanup() *commander.Command {
|
|
cmd := &commander.Command{
|
|
Run: aptlyDbCleanup,
|
|
UsageLine: "cleanup",
|
|
Short: "cleanup DB and package pool",
|
|
Long: `
|
|
Database cleanup removes information about unreferenced packages and removes
|
|
files in the package pool that aren't used by packages anymore
|
|
|
|
Example:
|
|
|
|
$ aptly db cleanup
|
|
`,
|
|
}
|
|
|
|
cmd.Flag.Bool("verbose", false, "be verbose when loading objects/removing them")
|
|
cmd.Flag.Bool("dry-run", false, "don't delete anything")
|
|
|
|
return cmd
|
|
}
|