Skip to content

Commit

Permalink
Merge branch 'main' into renovate/js-bufbuild-protoc-gen-es-2.x
Browse files Browse the repository at this point in the history
  • Loading branch information
wesbillman authored Oct 2, 2024
2 parents 0e8ed03 + 9479c62 commit 34a47cb
Show file tree
Hide file tree
Showing 311 changed files with 6,789 additions and 3,382 deletions.
2 changes: 1 addition & 1 deletion .go-arch-lint.yml
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ components:
internal: { in: internal/** }
dal: { in: backend/dal/** }
protos: { in: backend/protos/** }
schema: { in: backend/schema/** }
schema: { in: internal/schema/** }
ftl: { in: . }
ftl-cmd: { in: frontend/cli/ftl/** }
go-runtime: { in: go-runtime/** }
Expand Down
4 changes: 2 additions & 2 deletions Justfile
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,7 @@ generate-kube-migrations:

# Run "go generate" on all packages
build-generate:
@mk backend/schema/aliaskind_enumer.go : backend/schema/metadataalias.go -- go generate -x ./backend/schema
@mk internal/schema/aliaskind_enumer.go : internal/schema/metadataalias.go -- go generate -x ./internal/schema
@mk internal/log/log_level_string.go : internal/log/api.go -- go generate -x ./internal/log

# Build command-line tools
Expand Down Expand Up @@ -129,7 +129,7 @@ pnpm-install:

# Regenerate protos
build-protos: pnpm-install
@mk {{SCHEMA_OUT}} : backend/schema -- "ftl-schema > {{SCHEMA_OUT}} && buf format -w && buf lint"
@mk {{SCHEMA_OUT}} : internal/schema -- "ftl-schema > {{SCHEMA_OUT}} && buf format -w && buf lint"
@mk {{PROTOS_OUT}} : {{PROTOS_IN}} -- "cd backend/protos && buf generate"

# Unconditionally rebuild protos
Expand Down
2 changes: 1 addition & 1 deletion backend/controller/admin/admin.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,12 @@ import (

ftlv1 "github.com/TBD54566975/ftl/backend/protos/xyz/block/ftl/v1"
"github.com/TBD54566975/ftl/backend/protos/xyz/block/ftl/v1/ftlv1connect"
"github.com/TBD54566975/ftl/backend/schema"
"github.com/TBD54566975/ftl/go-runtime/encoding"
"github.com/TBD54566975/ftl/internal/configuration"
"github.com/TBD54566975/ftl/internal/configuration/manager"
"github.com/TBD54566975/ftl/internal/configuration/providers"
"github.com/TBD54566975/ftl/internal/log"
"github.com/TBD54566975/ftl/internal/schema"
)

type AdminService struct {
Expand Down
2 changes: 1 addition & 1 deletion backend/controller/admin/admin_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,12 +13,12 @@ import (
"github.com/alecthomas/types/optional"

ftlv1 "github.com/TBD54566975/ftl/backend/protos/xyz/block/ftl/v1"
"github.com/TBD54566975/ftl/backend/schema"
"github.com/TBD54566975/ftl/internal/configuration"
"github.com/TBD54566975/ftl/internal/configuration/manager"
"github.com/TBD54566975/ftl/internal/configuration/providers"
"github.com/TBD54566975/ftl/internal/configuration/routers"
"github.com/TBD54566975/ftl/internal/log"
"github.com/TBD54566975/ftl/internal/schema"
)

func TestAdminService(t *testing.T) {
Expand Down
6 changes: 3 additions & 3 deletions backend/controller/admin/local_client.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,11 @@ import (

"github.com/alecthomas/types/optional"

"github.com/TBD54566975/ftl/backend/schema"
"github.com/TBD54566975/ftl/internal/buildengine"
cf "github.com/TBD54566975/ftl/internal/configuration"
"github.com/TBD54566975/ftl/internal/configuration/manager"
"github.com/TBD54566975/ftl/internal/projectconfig"
"github.com/TBD54566975/ftl/internal/schema"
"github.com/TBD54566975/ftl/internal/watch"
)

// localClient reads and writes to local projectconfig files without making any network
Expand Down Expand Up @@ -40,7 +40,7 @@ func (s *diskSchemaRetriever) GetActiveSchema(ctx context.Context) (*schema.Sche
if err != nil {
return nil, fmt.Errorf("could not load project config: %w", err)
}
modules, err := buildengine.DiscoverModules(ctx, projConfig.AbsModuleDirs())
modules, err := watch.DiscoverModules(ctx, projConfig.AbsModuleDirs())
if err != nil {
return nil, fmt.Errorf("could not discover modules: %w", err)
}
Expand Down
2 changes: 1 addition & 1 deletion backend/controller/admin/testdata/go/dischema/go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ module ftl/dischema

go 1.23.0

require github.com/TBD54566975/ftl v1.1.5
require github.com/TBD54566975/ftl v0.0.0-00010101000000-000000000000

require (
connectrpc.com/connect v1.16.2 // indirect
Expand Down
45 changes: 45 additions & 0 deletions backend/controller/artefacts/artefact.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
package artefacts

import (
"context"
"io"

"github.com/TBD54566975/ftl/internal/model"
"github.com/TBD54566975/ftl/internal/sha256"
)

// Metadata container for an artefact's metadata
type Metadata struct {
Executable bool
Path string
}

// Artefact container for an artefact's payload and metadata
type Artefact struct {
Digest sha256.SHA256
Metadata Metadata
Content []byte
}

type ArtefactKey struct {
Digest sha256.SHA256
}

type ReleaseArtefact struct {
Artefact ArtefactKey
Path string
Executable bool
}

type Registry interface {
// GetDigestsKeys locates the `digests` corresponding `ArtefactKey`s and identifies the missing ones
GetDigestsKeys(ctx context.Context, digests []sha256.SHA256) (keys []ArtefactKey, missing []sha256.SHA256, err error)
// Upload pushes the specified media, and metadata, to the registry and returns the computed digest
Upload(context context.Context, artefact Artefact) (sha256.SHA256, error)
// Download performs a streaming download of the artefact identified by the supplied digest
Download(context context.Context, digest sha256.SHA256) (io.ReadCloser, error)
// GetReleaseArtefacts locates the artefacts metadata corresponding with the specified release
GetReleaseArtefacts(ctx context.Context, releaseID int64) ([]ArtefactKey, error)
// AddReleaseArtefact associates the given `release` with the artefact associated with the given `digest`
AddReleaseArtefact(ctx context.Context, key model.DeploymentKey, ra ReleaseArtefact) error
}
129 changes: 129 additions & 0 deletions backend/controller/artefacts/dal_registry.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,129 @@
package artefacts

import (
"context"
"fmt"
"io"

sets "github.com/deckarep/golang-set/v2"

"github.com/TBD54566975/ftl/backend/controller/artefacts/internal/sql"
"github.com/TBD54566975/ftl/backend/libdal"
"github.com/TBD54566975/ftl/internal/model"
"github.com/TBD54566975/ftl/internal/sha256"
"github.com/TBD54566975/ftl/internal/slices"
)

type ArtefactRow struct {
ID int64
Digest []byte
}

type DAL interface {
GetArtefactDigests(ctx context.Context, digests [][]byte) ([]ArtefactRow, error)
CreateArtefact(ctx context.Context, digest []byte, content []byte) (int64, error)
GetArtefactContentRange(ctx context.Context, start int32, count int32, iD int64) ([]byte, error)
}

type Service struct {
*libdal.Handle[Service]
db sql.Querier
}

func New(conn libdal.Connection) *Service {
return &Service{
db: sql.New(conn),
Handle: libdal.New(conn, func(h *libdal.Handle[Service]) *Service {
return &Service{
Handle: h,
db: sql.New(h.Connection),
}
}),
}
}

func (s *Service) GetDigestsKeys(ctx context.Context, digests []sha256.SHA256) (keys []ArtefactKey, missing []sha256.SHA256, err error) {
have, err := s.db.GetArtefactDigests(ctx, sha256esToBytes(digests))
if err != nil {
return nil, nil, libdal.TranslatePGError(err)
}
keys = slices.Map(have, func(in sql.GetArtefactDigestsRow) ArtefactKey {
return ArtefactKey{Digest: sha256.FromBytes(in.Digest)}
})
haveStr := slices.Map(keys, func(in ArtefactKey) sha256.SHA256 {
return in.Digest
})
missing = sets.NewSet(digests...).Difference(sets.NewSet(haveStr...)).ToSlice()
return keys, missing, nil
}

func (s *Service) Upload(ctx context.Context, artefact Artefact) (sha256.SHA256, error) {
sha256digest := sha256.Sum(artefact.Content)
_, err := s.db.CreateArtefact(ctx, sha256digest[:], artefact.Content)
return sha256digest, libdal.TranslatePGError(err)
}

func (s *Service) Download(ctx context.Context, digest sha256.SHA256) (io.ReadCloser, error) {
digests := [][]byte{digest[:]}
rows, err := s.db.GetArtefactDigests(ctx, digests)
if err != nil {
return nil, fmt.Errorf("unable to get artefact digests: %w", libdal.TranslatePGError(err))
}
if len(rows) == 0 {
return nil, fmt.Errorf("no artefact found with digest: %s", digest)
}
return &dalArtefactStream{db: s.db, id: rows[0].ID}, nil
}

func (s *Service) GetReleaseArtefacts(ctx context.Context, releaseID int64) ([]ReleaseArtefact, error) {
rows, err := s.db.GetDeploymentArtefacts(ctx, releaseID)
if err != nil {
return nil, fmt.Errorf("unable to get release artefacts: %w", libdal.TranslatePGError(err))
}
return slices.Map(rows, func(row sql.GetDeploymentArtefactsRow) ReleaseArtefact {
return ReleaseArtefact{
Artefact: ArtefactKey{Digest: sha256.FromBytes(row.Digest)},
Path: row.Path,
Executable: row.Executable,
}
}), nil
}

func (s *Service) AddReleaseArtefact(ctx context.Context, key model.DeploymentKey, ra ReleaseArtefact) error {
err := s.db.AssociateArtefactWithDeployment(ctx, sql.AssociateArtefactWithDeploymentParams{
Key: key,
Digest: ra.Artefact.Digest[:],
Executable: ra.Executable,
Path: ra.Path,
})
if err != nil {
return libdal.TranslatePGError(err)
}
return nil
}

type dalArtefactStream struct {
db sql.Querier
id int64
offset int32
}

func (s *dalArtefactStream) Close() error { return nil }

func (s *dalArtefactStream) Read(p []byte) (n int, err error) {
content, err := s.db.GetArtefactContentRange(context.Background(), s.offset+1, int32(len(p)), s.id)
if err != nil {
return 0, libdal.TranslatePGError(err)
}
copy(p, content)
clen := len(content)
s.offset += int32(clen)
if clen == 0 {
err = io.EOF
}
return clen, err
}

func sha256esToBytes(digests []sha256.SHA256) [][]byte {
return slices.Map(digests, func(digest sha256.SHA256) []byte { return digest[:] })
}
22 changes: 22 additions & 0 deletions backend/controller/artefacts/internal/sql/querier.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

29 changes: 29 additions & 0 deletions backend/controller/artefacts/internal/sql/queries.sql
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
-- name: GetArtefactDigests :many
-- Return the digests that exist in the database.
SELECT id, digest
FROM artefacts
WHERE digest = ANY (@digests::bytea[]);

-- name: CreateArtefact :one
-- Create a new artefact and return the artefact ID.
INSERT INTO artefacts (digest, content)
VALUES ($1, $2)
ON CONFLICT (digest)
DO UPDATE SET digest = EXCLUDED.digest
RETURNING id;

-- name: GetArtefactContentRange :one
SELECT SUBSTRING(a.content FROM @start FOR @count)::BYTEA AS content
FROM artefacts a
WHERE a.id = @id;

-- name: GetDeploymentArtefacts :many
-- Get all artefacts matching the given digests.
SELECT da.created_at, artefact_id AS id, executable, path, digest, executable
FROM deployment_artefacts da
INNER JOIN artefacts ON artefacts.id = da.artefact_id
WHERE deployment_id = $1;

-- name: AssociateArtefactWithDeployment :exec
INSERT INTO deployment_artefacts (deployment_id, artefact_id, executable, path)
VALUES ((SELECT id FROM deployments WHERE key = @key::deployment_key), (SELECT id FROM artefacts WHERE digest = @digest::bytea), $3, $4);
Loading

0 comments on commit 34a47cb

Please sign in to comment.