3dda067747
This integrates the new module, which was extracted from this repository at commit b9b19409cf458dcb9e1253ff44ba75bd0620faa6; # install filter-repo (https://github.com/newren/git-filter-repo/blob/main/INSTALL.md) brew install git-filter-repo # create a temporary clone of docker cd ~/Projects git clone https://github.com/distribution/distribution.git reference cd reference # commit taken from git rev-parse --verify HEADb9b19409cf
# remove all code, except for general files, 'reference/', and rename to / git filter-repo \ --path .github/workflows/codeql-analysis.yml \ --path .github/workflows/fossa.yml \ --path .golangci.yml \ --path distribution-logo.svg \ --path CODE-OF-CONDUCT.md \ --path CONTRIBUTING.md \ --path GOVERNANCE.md \ --path README.md \ --path LICENSE \ --path MAINTAINERS \ --path-glob 'reference/*.*' \ --path-rename reference/: # initialize go.mod go mod init github.com/distribution/reference go mod tidy -go=1.20 This commit is based on152af63ec5
in the main branch, but adjusted for the 2.8 branch, with some differences: - the Sort functions have not been kept, as they were not part of the v2 package, and introduced in1052518d9f
- the ParseAnyReferenceWithSet and ShortIdentifierRegexp were kept (but deprecated) as removing happened in6d4f62d7fd
, which is not in the 2.8 branch. Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
225 lines
6.2 KiB
Go
225 lines
6.2 KiB
Go
package notifications
|
|
|
|
import (
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/distribution/reference"
|
|
"github.com/docker/distribution"
|
|
"github.com/docker/distribution/context"
|
|
"github.com/docker/distribution/uuid"
|
|
"github.com/opencontainers/go-digest"
|
|
)
|
|
|
|
type bridge struct {
|
|
ub URLBuilder
|
|
includeReferences bool
|
|
actor ActorRecord
|
|
source SourceRecord
|
|
request RequestRecord
|
|
sink Sink
|
|
}
|
|
|
|
var _ Listener = &bridge{}
|
|
|
|
// URLBuilder defines a subset of url builder to be used by the event listener.
|
|
type URLBuilder interface {
|
|
BuildManifestURL(name reference.Named) (string, error)
|
|
BuildBlobURL(ref reference.Canonical) (string, error)
|
|
}
|
|
|
|
// NewBridge returns a notification listener that writes records to sink,
|
|
// using the actor and source. Any urls populated in the events created by
|
|
// this bridge will be created using the URLBuilder.
|
|
// TODO(stevvooe): Update this to simply take a context.Context object.
|
|
func NewBridge(ub URLBuilder, source SourceRecord, actor ActorRecord, request RequestRecord, sink Sink, includeReferences bool) Listener {
|
|
return &bridge{
|
|
ub: ub,
|
|
includeReferences: includeReferences,
|
|
actor: actor,
|
|
source: source,
|
|
request: request,
|
|
sink: sink,
|
|
}
|
|
}
|
|
|
|
// NewRequestRecord builds a RequestRecord for use in NewBridge from an
|
|
// http.Request, associating it with a request id.
|
|
func NewRequestRecord(id string, r *http.Request) RequestRecord {
|
|
return RequestRecord{
|
|
ID: id,
|
|
Addr: context.RemoteAddr(r),
|
|
Host: r.Host,
|
|
Method: r.Method,
|
|
UserAgent: r.UserAgent(),
|
|
}
|
|
}
|
|
|
|
func (b *bridge) ManifestPushed(repo reference.Named, sm distribution.Manifest, options ...distribution.ManifestServiceOption) error {
|
|
manifestEvent, err := b.createManifestEvent(EventActionPush, repo, sm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, option := range options {
|
|
if opt, ok := option.(distribution.WithTagOption); ok {
|
|
manifestEvent.Target.Tag = opt.Tag
|
|
break
|
|
}
|
|
}
|
|
return b.sink.Write(*manifestEvent)
|
|
}
|
|
|
|
func (b *bridge) ManifestPulled(repo reference.Named, sm distribution.Manifest, options ...distribution.ManifestServiceOption) error {
|
|
manifestEvent, err := b.createManifestEvent(EventActionPull, repo, sm)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, option := range options {
|
|
if opt, ok := option.(distribution.WithTagOption); ok {
|
|
manifestEvent.Target.Tag = opt.Tag
|
|
break
|
|
}
|
|
}
|
|
return b.sink.Write(*manifestEvent)
|
|
}
|
|
|
|
func (b *bridge) ManifestDeleted(repo reference.Named, dgst digest.Digest) error {
|
|
return b.createManifestDeleteEventAndWrite(EventActionDelete, repo, dgst)
|
|
}
|
|
|
|
func (b *bridge) BlobPushed(repo reference.Named, desc distribution.Descriptor) error {
|
|
return b.createBlobEventAndWrite(EventActionPush, repo, desc)
|
|
}
|
|
|
|
func (b *bridge) BlobPulled(repo reference.Named, desc distribution.Descriptor) error {
|
|
return b.createBlobEventAndWrite(EventActionPull, repo, desc)
|
|
}
|
|
|
|
func (b *bridge) BlobMounted(repo reference.Named, desc distribution.Descriptor, fromRepo reference.Named) error {
|
|
event, err := b.createBlobEvent(EventActionMount, repo, desc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
event.Target.FromRepository = fromRepo.Name()
|
|
return b.sink.Write(*event)
|
|
}
|
|
|
|
func (b *bridge) BlobDeleted(repo reference.Named, dgst digest.Digest) error {
|
|
return b.createBlobDeleteEventAndWrite(EventActionDelete, repo, dgst)
|
|
}
|
|
|
|
func (b *bridge) TagDeleted(repo reference.Named, tag string) error {
|
|
event := b.createEvent(EventActionDelete)
|
|
event.Target.Repository = repo.Name()
|
|
event.Target.Tag = tag
|
|
|
|
return b.sink.Write(*event)
|
|
}
|
|
|
|
func (b *bridge) RepoDeleted(repo reference.Named) error {
|
|
event := b.createEvent(EventActionDelete)
|
|
event.Target.Repository = repo.Name()
|
|
|
|
return b.sink.Write(*event)
|
|
}
|
|
|
|
func (b *bridge) createManifestDeleteEventAndWrite(action string, repo reference.Named, dgst digest.Digest) error {
|
|
event := b.createEvent(action)
|
|
event.Target.Repository = repo.Name()
|
|
event.Target.Digest = dgst
|
|
|
|
return b.sink.Write(*event)
|
|
}
|
|
|
|
func (b *bridge) createManifestEvent(action string, repo reference.Named, sm distribution.Manifest) (*Event, error) {
|
|
event := b.createEvent(action)
|
|
event.Target.Repository = repo.Name()
|
|
|
|
mt, p, err := sm.Payload()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Ensure we have the canonical manifest descriptor here
|
|
manifest, desc, err := distribution.UnmarshalManifest(mt, p)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
event.Target.MediaType = mt
|
|
event.Target.Length = desc.Size
|
|
event.Target.Size = desc.Size
|
|
event.Target.Digest = desc.Digest
|
|
if b.includeReferences {
|
|
event.Target.References = append(event.Target.References, manifest.References()...)
|
|
}
|
|
|
|
ref, err := reference.WithDigest(repo, event.Target.Digest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
event.Target.URL, err = b.ub.BuildManifestURL(ref)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return event, nil
|
|
}
|
|
|
|
func (b *bridge) createBlobDeleteEventAndWrite(action string, repo reference.Named, dgst digest.Digest) error {
|
|
event := b.createEvent(action)
|
|
event.Target.Digest = dgst
|
|
event.Target.Repository = repo.Name()
|
|
|
|
return b.sink.Write(*event)
|
|
}
|
|
|
|
func (b *bridge) createBlobEventAndWrite(action string, repo reference.Named, desc distribution.Descriptor) error {
|
|
event, err := b.createBlobEvent(action, repo, desc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return b.sink.Write(*event)
|
|
}
|
|
|
|
func (b *bridge) createBlobEvent(action string, repo reference.Named, desc distribution.Descriptor) (*Event, error) {
|
|
event := b.createEvent(action)
|
|
event.Target.Descriptor = desc
|
|
event.Target.Length = desc.Size
|
|
event.Target.Repository = repo.Name()
|
|
|
|
ref, err := reference.WithDigest(repo, desc.Digest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
event.Target.URL, err = b.ub.BuildBlobURL(ref)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return event, nil
|
|
}
|
|
|
|
// createEvent creates an event with actor and source populated.
|
|
func (b *bridge) createEvent(action string) *Event {
|
|
event := createEvent(action)
|
|
event.Source = b.source
|
|
event.Actor = b.actor
|
|
event.Request = b.request
|
|
|
|
return event
|
|
}
|
|
|
|
// createEvent returns a new event, timestamped, with the specified action.
|
|
func createEvent(action string) *Event {
|
|
return &Event{
|
|
ID: uuid.Generate().String(),
|
|
Timestamp: time.Now(),
|
|
Action: action,
|
|
}
|
|
}
|