|
| 1 | +package compose |
| 2 | + |
| 3 | +import ( |
| 4 | + "context" |
| 5 | + "fmt" |
| 6 | + "strings" |
| 7 | + |
| 8 | + "dagger.io/dagger" |
| 9 | + "dagger.io/dagger/dag" |
| 10 | + "dagger.io/dockersdk/module/object" |
| 11 | + "dagger.io/dockersdk/module/proxy" |
| 12 | +) |
| 13 | + |
| 14 | +// allFunc is a function that starts all services using |
| 15 | +// a proxy module to group them together. |
| 16 | +// |
| 17 | +// It MUST be registered AFTER all services have been registered. |
| 18 | +// The proxy is a simple duplication of: github.com/kpenfound/dagger-modules/proxy@v0.2.5 module. |
| 19 | +type allFunc struct { |
| 20 | + c *Compose |
| 21 | +} |
| 22 | + |
| 23 | +func (u *allFunc) up(services []*proxy.Service) *dagger.Container { |
| 24 | + proxy := proxy.New() |
| 25 | + |
| 26 | + for _, service := range services { |
| 27 | + proxy = proxy.WithService(service) |
| 28 | + } |
| 29 | + |
| 30 | + return proxy.Service() |
| 31 | +} |
| 32 | + |
| 33 | +func (u *allFunc) Invoke(ctx context.Context, state object.State, input object.InputArgs) (object.Result, error) { |
| 34 | + compose, err := u.c.load(state) |
| 35 | + if err != nil { |
| 36 | + return nil, fmt.Errorf("failed to load object state: %w", err) |
| 37 | + } |
| 38 | + |
| 39 | + services := []*proxy.Service{} |
| 40 | + for _, service := range u.c.dockercompose.Services() { |
| 41 | + service := &serviceFunc{c: compose, service: service} |
| 42 | + servicePrefix := fmt.Sprintf("%s_", service.service.Name()) |
| 43 | + |
| 44 | + serviceInput := input |
| 45 | + for argName, argValue := range input { |
| 46 | + if strings.HasPrefix(argName, servicePrefix) { |
| 47 | + serviceInput[strings.TrimPrefix(argName, servicePrefix)] = argValue |
| 48 | + } |
| 49 | + } |
| 50 | + |
| 51 | + serviceCtr, err := service.ToService(ctx, state, serviceInput) |
| 52 | + if err != nil { |
| 53 | + return nil, fmt.Errorf("failed to get service %s: %w", service.service.Name(), err) |
| 54 | + } |
| 55 | + |
| 56 | + services = append(services, serviceCtr) |
| 57 | + } |
| 58 | + |
| 59 | + return (*allFunc).up( |
| 60 | + &allFunc{c: compose}, |
| 61 | + services, |
| 62 | + ), nil |
| 63 | +} |
| 64 | + |
| 65 | +func (u *allFunc) Arguments() []*object.FunctionArg { |
| 66 | + // This method should not be called for this function |
| 67 | + return nil |
| 68 | +} |
| 69 | + |
| 70 | +func (u *allFunc) AddTypeDefToObject(ctx context.Context, mod *dagger.Module, obj *dagger.TypeDef) (*dagger.Module, *dagger.TypeDef) { |
| 71 | + args := []*object.FunctionArg{} |
| 72 | + |
| 73 | + serviceNames := []string{} |
| 74 | + for name, service := range u.c.funcMap { |
| 75 | + serviceNames = append(serviceNames, name) |
| 76 | + |
| 77 | + serviceArgs := service.Arguments() |
| 78 | + for _, arg := range serviceArgs { |
| 79 | + args = append(args, &object.FunctionArg{ |
| 80 | + // Prefix the argument name with the service name to avoid colission |
| 81 | + Name: fmt.Sprintf("%s_%s", name, arg.Name), |
| 82 | + Type: arg.Type, |
| 83 | + Opts: arg.Opts, |
| 84 | + }) |
| 85 | + } |
| 86 | + } |
| 87 | + |
| 88 | + typedef := dag.Function("All", dag.TypeDef().WithObject("Container")). |
| 89 | + WithDescription(fmt.Sprintf("Start all service containers (%s)", strings.Join(serviceNames, ", "))) |
| 90 | + |
| 91 | + for _, arg := range args { |
| 92 | + typedef = typedef.WithArg(arg.Name, arg.Type, arg.Opts) |
| 93 | + } |
| 94 | + |
| 95 | + return mod, obj. |
| 96 | + WithFunction(typedef) |
| 97 | +} |
0 commit comments