mirror of
				https://github.com/containers/podman.git
				synced 2025-10-27 03:06:22 +08:00 
			
		
		
		
	 b90f7f9095
			
		
	
	b90f7f9095
	
	
	
		
			
			This change adds code to report the reclaimed space after a prune. Reclaimed space from volumes, images, and containers is recorded during the prune call in a PruneReport struct. These structs are collected into a slice during a system prune and processed afterwards to calculate the total reclaimed space. Closes #8658 Signed-off-by: Baron Lenardson <lenardson.baron@gmail.com>
		
			
				
	
	
		
			166 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			166 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package libpod
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 
 | |
| 	"github.com/containers/podman/v2/libpod/define"
 | |
| 	"github.com/containers/podman/v2/libpod/events"
 | |
| 	"github.com/containers/podman/v2/pkg/domain/entities/reports"
 | |
| 	"github.com/pkg/errors"
 | |
| )
 | |
| 
 | |
| // Contains the public Runtime API for volumes
 | |
| 
 | |
| // A VolumeCreateOption is a functional option which alters the Volume created by
 | |
| // NewVolume
 | |
| type VolumeCreateOption func(*Volume) error
 | |
| 
 | |
| // VolumeFilter is a function to determine whether a volume is included in command
 | |
| // output. Volumes to be outputted are tested using the function. a true return will
 | |
| // include the volume, a false return will exclude it.
 | |
| type VolumeFilter func(*Volume) bool
 | |
| 
 | |
| // RemoveVolume removes a volumes
 | |
| func (r *Runtime) RemoveVolume(ctx context.Context, v *Volume, force bool) error {
 | |
| 	r.lock.Lock()
 | |
| 	defer r.lock.Unlock()
 | |
| 
 | |
| 	if !r.valid {
 | |
| 		return define.ErrRuntimeStopped
 | |
| 	}
 | |
| 
 | |
| 	if !v.valid {
 | |
| 		if ok, _ := r.state.HasVolume(v.Name()); !ok {
 | |
| 			// Volume probably already removed
 | |
| 			// Or was never in the runtime to begin with
 | |
| 			return nil
 | |
| 		}
 | |
| 	}
 | |
| 	return r.removeVolume(ctx, v, force)
 | |
| }
 | |
| 
 | |
| // GetVolume retrieves a volume given its full name.
 | |
| func (r *Runtime) GetVolume(name string) (*Volume, error) {
 | |
| 	r.lock.RLock()
 | |
| 	defer r.lock.RUnlock()
 | |
| 
 | |
| 	if !r.valid {
 | |
| 		return nil, define.ErrRuntimeStopped
 | |
| 	}
 | |
| 
 | |
| 	vol, err := r.state.Volume(name)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return vol, nil
 | |
| }
 | |
| 
 | |
| // LookupVolume retrieves a volume by unambiguous partial name.
 | |
| func (r *Runtime) LookupVolume(name string) (*Volume, error) {
 | |
| 	r.lock.RLock()
 | |
| 	defer r.lock.RUnlock()
 | |
| 
 | |
| 	if !r.valid {
 | |
| 		return nil, define.ErrRuntimeStopped
 | |
| 	}
 | |
| 
 | |
| 	vol, err := r.state.LookupVolume(name)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	return vol, nil
 | |
| }
 | |
| 
 | |
| // HasVolume checks to see if a volume with the given name exists
 | |
| func (r *Runtime) HasVolume(name string) (bool, error) {
 | |
| 	r.lock.RLock()
 | |
| 	defer r.lock.RUnlock()
 | |
| 
 | |
| 	if !r.valid {
 | |
| 		return false, define.ErrRuntimeStopped
 | |
| 	}
 | |
| 
 | |
| 	return r.state.HasVolume(name)
 | |
| }
 | |
| 
 | |
| // Volumes retrieves all volumes
 | |
| // Filters can be provided which will determine which volumes are included in the
 | |
| // output. If multiple filters are used, a volume will be returned if
 | |
| // any of the filters are matched
 | |
| func (r *Runtime) Volumes(filters ...VolumeFilter) ([]*Volume, error) {
 | |
| 	r.lock.RLock()
 | |
| 	defer r.lock.RUnlock()
 | |
| 
 | |
| 	if !r.valid {
 | |
| 		return nil, define.ErrRuntimeStopped
 | |
| 	}
 | |
| 
 | |
| 	vols, err := r.state.AllVolumes()
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	if len(filters) == 0 {
 | |
| 		return vols, nil
 | |
| 	}
 | |
| 
 | |
| 	volsFiltered := make([]*Volume, 0, len(vols))
 | |
| 	for _, vol := range vols {
 | |
| 		include := false
 | |
| 		for _, filter := range filters {
 | |
| 			include = include || filter(vol)
 | |
| 		}
 | |
| 
 | |
| 		if include {
 | |
| 			volsFiltered = append(volsFiltered, vol)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return volsFiltered, nil
 | |
| }
 | |
| 
 | |
| // GetAllVolumes retrieves all the volumes
 | |
| func (r *Runtime) GetAllVolumes() ([]*Volume, error) {
 | |
| 	r.lock.RLock()
 | |
| 	defer r.lock.RUnlock()
 | |
| 
 | |
| 	if !r.valid {
 | |
| 		return nil, define.ErrRuntimeStopped
 | |
| 	}
 | |
| 
 | |
| 	return r.state.AllVolumes()
 | |
| }
 | |
| 
 | |
| // PruneVolumes removes unused volumes from the system
 | |
| func (r *Runtime) PruneVolumes(ctx context.Context, filterFuncs []VolumeFilter) ([]*reports.PruneReport, error) {
 | |
| 	preports := make([]*reports.PruneReport, 0)
 | |
| 	vols, err := r.Volumes(filterFuncs...)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	for _, vol := range vols {
 | |
| 		report := new(reports.PruneReport)
 | |
| 		volSize, err := vol.Size()
 | |
| 		if err != nil {
 | |
| 			volSize = 0
 | |
| 		}
 | |
| 		report.Size = volSize
 | |
| 		report.Id = vol.Name()
 | |
| 		if err := r.RemoveVolume(ctx, vol, false); err != nil {
 | |
| 			if errors.Cause(err) != define.ErrVolumeBeingUsed && errors.Cause(err) != define.ErrVolumeRemoved {
 | |
| 				report.Err = err
 | |
| 			} else {
 | |
| 				// We didn't remove the volume for some reason
 | |
| 				continue
 | |
| 			}
 | |
| 		} else {
 | |
| 			vol.newVolumeEvent(events.Prune)
 | |
| 		}
 | |
| 		preports = append(preports, report)
 | |
| 	}
 | |
| 	return preports, nil
 | |
| }
 |