europa: Code Embedding POC
WIP PoC for code embedding Signed-off-by: Andrea Luzzardi <aluzzardi@gmail.com>
This commit is contained in:
parent
63d655d8f3
commit
6abe77729e
@ -1,5 +1,20 @@
|
||||
package engine
|
||||
|
||||
// Access the source directory for the current CUE package
|
||||
// This may safely be called from any package
|
||||
#Source: {
|
||||
$dagger: task: _name: "Source"
|
||||
|
||||
// Relative path to source.
|
||||
path: string
|
||||
// Optionally exclude certain files
|
||||
include: [...string]
|
||||
// Optionall include certain files
|
||||
exclude: [...string]
|
||||
|
||||
output: #FS
|
||||
}
|
||||
|
||||
// Create one or multiple directory in a container
|
||||
#Mkdir: {
|
||||
$dagger: task: _name: "Mkdir"
|
||||
|
101
plan/task/source.go
Normal file
101
plan/task/source.go
Normal file
@ -0,0 +1,101 @@
|
||||
package task
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io/fs"
|
||||
"os"
|
||||
|
||||
"github.com/moby/buildkit/client/llb"
|
||||
"github.com/rs/zerolog/log"
|
||||
"go.dagger.io/dagger/compiler"
|
||||
"go.dagger.io/dagger/plancontext"
|
||||
"go.dagger.io/dagger/solver"
|
||||
)
|
||||
|
||||
func init() {
|
||||
Register("Source", func() Task { return &sourceTask{} })
|
||||
}
|
||||
|
||||
type sourceTask struct {
|
||||
}
|
||||
|
||||
func (c *sourceTask) PreRun(ctx context.Context, pctx *plancontext.Context, v *compiler.Value) error {
|
||||
path, err := v.Lookup("path").String()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if !fs.ValidPath(path) {
|
||||
return fmt.Errorf("invalid path %q", path)
|
||||
}
|
||||
|
||||
absPath, err := v.Lookup("path").AbsPath()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if _, err := os.Stat(path); errors.Is(err, os.ErrNotExist) {
|
||||
return fmt.Errorf("path %q does not exist", path)
|
||||
}
|
||||
|
||||
pctx.LocalDirs.Add(absPath)
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (c *sourceTask) Run(ctx context.Context, pctx *plancontext.Context, s solver.Solver, v *compiler.Value) (*compiler.Value, error) {
|
||||
lg := log.Ctx(ctx)
|
||||
|
||||
path, err := v.Lookup("path").AbsPath()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var source struct {
|
||||
Include []string
|
||||
Exclude []string
|
||||
}
|
||||
|
||||
if err := v.Decode(&source); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
lg.Debug().Str("path", path).Msg("loading local directory")
|
||||
opts := []llb.LocalOption{
|
||||
withCustomName(v, "Embed %s", path),
|
||||
llb.IncludePatterns(source.Include),
|
||||
llb.ExcludePatterns(source.Exclude),
|
||||
// Without hint, multiple `llb.Local` operations on the
|
||||
// same path get a different digest.
|
||||
llb.SessionID(s.SessionID()),
|
||||
llb.SharedKeyHint(path),
|
||||
}
|
||||
|
||||
// FIXME: Remove the `Copy` and use `Local` directly.
|
||||
//
|
||||
// Copy'ing is a costly operation which should be unnecessary.
|
||||
// However, using llb.Local directly breaks caching sometimes for unknown reasons.
|
||||
st := llb.Scratch().File(
|
||||
llb.Copy(
|
||||
llb.Local(
|
||||
path,
|
||||
opts...,
|
||||
),
|
||||
"/",
|
||||
"/",
|
||||
),
|
||||
withCustomName(v, "Embed %s [copy]", path),
|
||||
)
|
||||
|
||||
result, err := s.Solve(ctx, st, pctx.Platform.Get())
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
fs := pctx.FS.New(result)
|
||||
return compiler.NewValue().FillFields(map[string]interface{}{
|
||||
"output": fs.MarshalCUE(),
|
||||
})
|
||||
}
|
@ -132,3 +132,15 @@ setup() {
|
||||
|
||||
"$DAGGER" --europa up ./newsecret.cue
|
||||
}
|
||||
|
||||
@test "task: #Source" {
|
||||
cd "$TESTDIR"/tasks/source
|
||||
"$DAGGER" --europa up ./source.cue
|
||||
"$DAGGER" --europa up ./source_include_exclude.cue
|
||||
|
||||
run "$DAGGER" --europa up ./source_invalid_path.cue
|
||||
assert_failure
|
||||
|
||||
run "$DAGGER" --europa up ./source_not_exist.cue
|
||||
assert_failure
|
||||
}
|
||||
|
1
tests/tasks/source/hello.txt
Normal file
1
tests/tasks/source/hello.txt
Normal file
@ -0,0 +1 @@
|
||||
hello
|
42
tests/tasks/source/source.cue
Normal file
42
tests/tasks/source/source.cue
Normal file
@ -0,0 +1,42 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"dagger.io/dagger/engine"
|
||||
)
|
||||
|
||||
engine.#Plan & {
|
||||
actions: {
|
||||
image: engine.#Pull & {
|
||||
source: "alpine:3.15.0@sha256:e7d88de73db3d3fd9b2d63aa7f447a10fd0220b7cbf39803c803f2af9ba256b3"
|
||||
}
|
||||
|
||||
source: engine.#Source & {
|
||||
path: "."
|
||||
}
|
||||
|
||||
exec: engine.#Exec & {
|
||||
input: image.output
|
||||
mounts: code: {
|
||||
dest: "/src"
|
||||
contents: source.output
|
||||
}
|
||||
args: ["/src/test.sh"]
|
||||
}
|
||||
|
||||
verifyHello: engine.#ReadFile & {
|
||||
input: source.output
|
||||
path: "/hello.txt"
|
||||
} & {
|
||||
// assert result
|
||||
contents: "hello\n"
|
||||
}
|
||||
|
||||
verifyWorld: engine.#ReadFile & {
|
||||
input: source.output
|
||||
path: "/world.txt"
|
||||
} & {
|
||||
// assert result
|
||||
contents: "world\n"
|
||||
}
|
||||
}
|
||||
}
|
45
tests/tasks/source/source_include_exclude.cue
Normal file
45
tests/tasks/source/source_include_exclude.cue
Normal file
@ -0,0 +1,45 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"dagger.io/dagger/engine"
|
||||
)
|
||||
|
||||
engine.#Plan & {
|
||||
actions: {
|
||||
image: engine.#Pull & {
|
||||
source: "alpine:3.15.0@sha256:e7d88de73db3d3fd9b2d63aa7f447a10fd0220b7cbf39803c803f2af9ba256b3"
|
||||
}
|
||||
|
||||
sourceInclude: engine.#Source & {
|
||||
path: "."
|
||||
include: ["hello.txt"]
|
||||
}
|
||||
|
||||
sourceExclude: engine.#Source & {
|
||||
path: "."
|
||||
exclude: ["hello.txt"]
|
||||
}
|
||||
|
||||
test: engine.#Exec & {
|
||||
input: image.output
|
||||
mounts: {
|
||||
include: {
|
||||
dest: "/include"
|
||||
contents: sourceInclude.output
|
||||
}
|
||||
exclude: {
|
||||
dest: "/exclude"
|
||||
contents: sourceExclude.output
|
||||
}
|
||||
}
|
||||
|
||||
args: ["sh", "-c",
|
||||
#"""
|
||||
test "$(find /include/ | wc -l)" -eq 1
|
||||
test -f /include/hello.txt
|
||||
test ! -f /exclude/hello.txt
|
||||
"""#,
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
17
tests/tasks/source/source_invalid_path.cue
Normal file
17
tests/tasks/source/source_invalid_path.cue
Normal file
@ -0,0 +1,17 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"dagger.io/dagger/engine"
|
||||
)
|
||||
|
||||
engine.#Plan & {
|
||||
actions: {
|
||||
image: engine.#Pull & {
|
||||
source: "alpine:3.15.0@sha256:e7d88de73db3d3fd9b2d63aa7f447a10fd0220b7cbf39803c803f2af9ba256b3"
|
||||
}
|
||||
|
||||
source: engine.#Source & {
|
||||
path: ".."
|
||||
}
|
||||
}
|
||||
}
|
17
tests/tasks/source/source_not_exist.cue
Normal file
17
tests/tasks/source/source_not_exist.cue
Normal file
@ -0,0 +1,17 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"dagger.io/dagger/engine"
|
||||
)
|
||||
|
||||
engine.#Plan & {
|
||||
actions: {
|
||||
image: engine.#Pull & {
|
||||
source: "alpine:3.15.0@sha256:e7d88de73db3d3fd9b2d63aa7f447a10fd0220b7cbf39803c803f2af9ba256b3"
|
||||
}
|
||||
|
||||
source: engine.#Source & {
|
||||
path: "not/exist"
|
||||
}
|
||||
}
|
||||
}
|
3
tests/tasks/source/test.sh
Executable file
3
tests/tasks/source/test.sh
Executable file
@ -0,0 +1,3 @@
|
||||
#!/bin/sh
|
||||
|
||||
echo -n hello world > /test.txt
|
1
tests/tasks/source/world.txt
Normal file
1
tests/tasks/source/world.txt
Normal file
@ -0,0 +1 @@
|
||||
world
|
Reference in New Issue
Block a user