peridot/peridot/yumrepofs/v1/blob.go

155 lines
4.7 KiB
Go

// Copyright (c) All respective contributors to the Peridot Project. All rights reserved.
// Copyright (c) 2021-2022 Rocky Enterprise Software Foundation, Inc. All rights reserved.
// Copyright (c) 2021-2022 Ctrl IQ, Inc. All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its contributors
// may be used to endorse or promote products derived from this software without
// specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
package yumrepofsv1
import (
"bytes"
"compress/gzip"
"context"
"encoding/base64"
"fmt"
"github.com/aws/aws-sdk-go/aws"
"github.com/aws/aws-sdk-go/service/s3"
"github.com/spf13/viper"
"google.golang.org/genproto/googleapis/api/httpbody"
"google.golang.org/grpc"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/metadata"
"google.golang.org/grpc/status"
"io/ioutil"
yumrepofspb "peridot.resf.org/peridot/yumrepofs/pb"
"peridot.resf.org/utils"
"regexp"
"strings"
"time"
)
var (
ErrCouldNotFindRevision = status.Error(codes.NotFound, "revision not found")
ErrInvalidBlob = status.Error(codes.InvalidArgument, "invalid blob given")
RegexBlob = regexp.MustCompile(`^(.+)-(.+)(\.xml|\.xml\.gz|\.yaml|\.yaml\.gz)$`)
)
func (s *Server) GetBlob(ctx context.Context, req *yumrepofspb.GetBlobRequest) (*httpbody.HttpBody, error) {
if err := req.ValidateAll(); err != nil {
return nil, err
}
if req.Arch == "i386" {
req.Arch = "i686"
}
if strings.HasSuffix(req.Blob, ".sqlite.gz") {
s3Req, _ := s.s3.GetObjectRequest(&s3.GetObjectInput{
Bucket: aws.String(viper.GetString("s3-bucket")),
Key: utils.StringP(fmt.Sprintf("sqlite-files/%s", req.Blob)),
})
urlStr, err := s3Req.Presign(24 * time.Hour)
if err != nil {
s.log.Errorf("failed to presign s3 request: %v", err)
return nil, status.Error(codes.Internal, "failed to presign s3 request")
}
header := metadata.Pairs("Location", urlStr)
err = grpc.SetHeader(ctx, header)
if err != nil {
s.log.Errorf("failed to send header: %v", err)
return nil, status.Error(codes.Internal, "failed to send header")
}
return &httpbody.HttpBody{
ContentType: "text/plain",
Data: []byte(fmt.Sprintf("Redirecting to %s", urlStr)),
}, nil
}
if !RegexBlob.MatchString(req.Blob) {
return nil, ErrInvalidBlob
}
blob := RegexBlob.FindStringSubmatch(req.Blob)
revision, err := s.db.GetRepositoryRevision(blob[1])
if err != nil {
return nil, ErrCouldNotFindRevision
}
var dataB64 string
switch blob[2] {
case "PRIMARY":
dataB64 = revision.PrimaryXml
case "FILELISTS":
dataB64 = revision.FilelistsXml
case "OTHER":
dataB64 = revision.OtherXml
case "GROUPS":
dataB64 = revision.GroupsXml
case "MODULES":
dataB64 = revision.ModulesYaml
case "UPDATEINFO":
dataB64 = revision.UpdateinfoXml
default:
return nil, ErrInvalidBlob
}
contentType := "application/xml+gzip"
data, err := base64.StdEncoding.DecodeString(dataB64)
if err != nil {
return nil, err
}
if blob[3] == ".yaml.gz" {
contentType = "text/yaml+gzip"
}
if blob[3] == ".xml" || blob[3] == ".yaml" {
contentType = "application/xml"
if blob[3] == ".yaml" {
contentType = "text/yaml"
}
var buf bytes.Buffer
buf.Write(data)
r, err := gzip.NewReader(&buf)
if err != nil {
return nil, utils.InternalError
}
data, err = ioutil.ReadAll(r)
if err != nil {
return nil, utils.InternalError
}
}
return &httpbody.HttpBody{
ContentType: contentType,
Data: data,
}, nil
}