drone/registry/app/pkg/rpm/local.go
Tudor Macari a3d828f0a2 feat: [AH-1060]: RPM flows skeleton, RPM package upload (#3599)
* resolve PR comments
* fix lint
* resolve PR comments
* fix lint issue
* resolve PR comments
* resolve PR comments
* fix lint issue
* feat: [AH-1060]: RPM flows skeleton, RPM package upload/install
2025-04-16 06:05:50 +00:00

180 lines
5.2 KiB
Go

// Copyright 2023 Harness, Inc.
//
// Licensed 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 rpm
import (
"context"
"fmt"
"io"
"mime/multipart"
"net/http"
urlprovider "github.com/harness/gitness/app/url"
"github.com/harness/gitness/registry/app/api/openapi/contracts/artifact"
rpmmetadata "github.com/harness/gitness/registry/app/metadata/rpm"
"github.com/harness/gitness/registry/app/pkg"
"github.com/harness/gitness/registry/app/pkg/base"
"github.com/harness/gitness/registry/app/pkg/commons"
"github.com/harness/gitness/registry/app/pkg/filemanager"
rpmtype "github.com/harness/gitness/registry/app/pkg/types/rpm"
"github.com/harness/gitness/registry/app/storage"
"github.com/harness/gitness/registry/app/store"
"github.com/harness/gitness/store/database/dbtx"
"github.com/rs/zerolog/log"
)
var _ pkg.Artifact = (*localRegistry)(nil)
var _ Registry = (*localRegistry)(nil)
type localRegistry struct {
localBase base.LocalBase
fileManager filemanager.FileManager
proxyStore store.UpstreamProxyConfigRepository
tx dbtx.Transactor
registryDao store.RegistryRepository
imageDao store.ImageRepository
artifactDao store.ArtifactRepository
urlProvider urlprovider.Provider
localRegistryHelper LocalRegistryHelper
}
type LocalRegistry interface {
Registry
}
func NewLocalRegistry(
localBase base.LocalBase,
fileManager filemanager.FileManager,
proxyStore store.UpstreamProxyConfigRepository,
tx dbtx.Transactor,
registryDao store.RegistryRepository,
imageDao store.ImageRepository,
artifactDao store.ArtifactRepository,
urlProvider urlprovider.Provider,
localRegistryHelper LocalRegistryHelper,
) LocalRegistry {
return &localRegistry{
localBase: localBase,
fileManager: fileManager,
proxyStore: proxyStore,
tx: tx,
registryDao: registryDao,
imageDao: imageDao,
artifactDao: artifactDao,
urlProvider: urlProvider,
localRegistryHelper: localRegistryHelper,
}
}
func (c *localRegistry) GetArtifactType() artifact.RegistryType {
return artifact.RegistryTypeVIRTUAL
}
func (c *localRegistry) GetPackageTypes() []artifact.PackageType {
return []artifact.PackageType{artifact.PackageTypeRPM}
}
func (c *localRegistry) UploadPackageFile(
ctx context.Context,
info rpmtype.ArtifactInfo,
file multipart.File,
) (headers *commons.ResponseHeaders, sha256 string, err error) {
buf, err := CreateHashedBufferFromReader(file)
if err != nil {
return nil, "", err
}
defer buf.Close()
pkg, err := parsePackage(buf)
if err != nil {
log.Printf("failded to parse rpm package: %v", err)
return nil, "", err
}
if _, err := buf.Seek(0, io.SeekStart); err != nil {
return nil, "", err
}
info.Image = pkg.Name
info.Version = pkg.Version + "." + pkg.FileMetadata.Architecture
info.Metadata = rpmmetadata.Metadata{
VersionMetadata: *pkg.VersionMetadata,
FileMetadata: *pkg.FileMetadata,
}
fileName := fmt.Sprintf("%s-%s.%s.rpm", pkg.Name, pkg.Version, pkg.FileMetadata.Architecture)
if info.FileName == "" {
info.FileName = fileName
}
path := fmt.Sprintf("%s/%s/%s/%s", pkg.Name, pkg.Version, pkg.FileMetadata.Architecture, fileName)
rs, sha256, err := c.localBase.Upload(ctx, info.ArtifactInfo, fileName, info.Version, path, buf,
&rpmmetadata.RpmMetadata{
Metadata: info.Metadata,
})
if err != nil {
return nil, "", err
}
//TODO: make it async / atomic operation, implement artifact status (sync successful, sync failed..... statuses)
err = c.localRegistryHelper.BuildRegistryFiles(ctx, info)
if err != nil {
return nil, "", err
}
return rs, sha256, err
}
func (c *localRegistry) GetRepoData(
ctx context.Context,
info rpmtype.ArtifactInfo,
fileName string,
) (*commons.ResponseHeaders,
*storage.FileReader,
io.ReadCloser,
string,
error,
) {
responseHeaders := &commons.ResponseHeaders{
Headers: make(map[string]string),
Code: 0,
}
fileReader, _, redirectURL, err := c.fileManager.DownloadFile(
ctx, "/"+RepoDataPrefix+fileName, info.RegistryID, info.RegIdentifier, info.RootIdentifier,
)
if err != nil {
return responseHeaders, nil, nil, "", err
}
responseHeaders.Code = http.StatusOK
return responseHeaders, fileReader, nil, redirectURL, nil
}
func (c *localRegistry) DownloadPackageFile(
ctx context.Context,
info rpmtype.ArtifactInfo,
) (*commons.ResponseHeaders, *storage.FileReader, io.ReadCloser, string, error) {
headers, fileReader, redirectURL, err := c.localBase.Download(
ctx, info.ArtifactInfo,
fmt.Sprintf("%s/%s", info.Version, info.Arch),
info.FileName,
)
if err != nil {
return nil, nil, nil, "", err
}
return headers, fileReader, nil, redirectURL, nil
}