Files
athens/pkg/storage/module/upload.go
Nicholas Wiersma d932d50232 chore: lint code with golangci-lint (#1828)
* feat: add golangci-lint linting

* chore: fix linter issues

* feat: add linting into the workflow

* docs: update lint docs

* fix: cr suggestions

* fix: remove old formatting and vetting scripts

* fix: add docker make target

* fix: action go caching

* fix: depreciated actions checkout version

* fix: cr suggestion

* fix: cr suggestions

---------

Co-authored-by: Manu Gupta <manugupt1@gmail.com>
2023-02-24 20:39:17 -08:00

64 lines
1.6 KiB
Go

package module
import (
"context"
"fmt"
"io"
"time"
"github.com/gomods/athens/pkg/config"
"github.com/gomods/athens/pkg/errors"
multierror "github.com/hashicorp/go-multierror"
)
const numFiles = 3
// Uploader takes a stream and saves it to the blob store under a given path.
type Uploader func(ctx context.Context, path, contentType string, stream io.Reader) error
// Upload saves .info, .mod and .zip files to the blob store in parallel.
// Returns multierror containing errors from all uploads and timeouts.
func Upload(ctx context.Context, module, version string, info, mod, zip io.Reader, uploader Uploader, timeout time.Duration) error {
const op errors.Op = "module.Upload"
tctx, cancel := context.WithTimeout(ctx, timeout)
defer cancel()
save := func(ext, contentType string, stream io.Reader) <-chan error {
ec := make(chan error)
go func() {
defer close(ec)
p := config.PackageVersionedName(module, version, ext)
ec <- uploader(tctx, p, contentType, stream)
}()
return ec
}
errChan := make(chan error, numFiles)
saveOrAbort := func(ext, contentType string, stream io.Reader) {
select {
case err := <-save(ext, contentType, stream):
errChan <- err
case <-tctx.Done():
errChan <- fmt.Errorf("uploading %s.%s.%s failed: %w", module, version, ext, tctx.Err())
}
}
go saveOrAbort("info", "application/json", info)
go saveOrAbort("mod", "text/plain", mod)
go saveOrAbort("zip", "application/octet-stream", zip)
var errs error
for i := 0; i < numFiles; i++ {
err := <-errChan
if err != nil {
errs = multierror.Append(errs, err)
}
}
close(errChan)
if errs != nil {
return errors.E(op, errs)
}
return nil
}