2018-07-22 22:54:44 +02:00
|
|
|
package taskfile
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2018-09-10 03:29:29 +02:00
|
|
|
"strings"
|
2018-07-22 22:54:44 +02:00
|
|
|
)
|
|
|
|
|
2018-09-10 03:29:29 +02:00
|
|
|
// NamespaceSeparator contains the character that separates namescapes
|
|
|
|
const NamespaceSeparator = ":"
|
|
|
|
|
2018-07-22 22:54:44 +02:00
|
|
|
// Merge merges the second Taskfile into the first
|
2022-10-02 07:07:58 +02:00
|
|
|
func Merge(t1, t2 *Taskfile, includedTaskfile *IncludedTaskfile, namespaces ...string) error {
|
2018-07-22 22:54:44 +02:00
|
|
|
if t1.Version != t2.Version {
|
2022-05-15 02:00:15 +02:00
|
|
|
return fmt.Errorf(`task: Taskfiles versions should match. First is "%s" but second is "%s"`, t1.Version, t2.Version)
|
2018-07-22 22:54:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if t2.Expansions != 0 && t2.Expansions != 2 {
|
|
|
|
t1.Expansions = t2.Expansions
|
|
|
|
}
|
2022-01-14 02:11:47 +02:00
|
|
|
if t2.Output.IsSet() {
|
2018-07-22 22:54:44 +02:00
|
|
|
t1.Output = t2.Output
|
|
|
|
}
|
2018-12-02 18:17:32 +02:00
|
|
|
|
|
|
|
if t1.Includes == nil {
|
2021-01-01 23:27:50 +02:00
|
|
|
t1.Includes = &IncludedTaskfiles{}
|
2018-09-10 03:29:29 +02:00
|
|
|
}
|
2021-01-01 23:27:50 +02:00
|
|
|
t1.Includes.Merge(t2.Includes)
|
2018-12-02 18:17:32 +02:00
|
|
|
|
|
|
|
if t1.Vars == nil {
|
2020-03-29 21:54:59 +02:00
|
|
|
t1.Vars = &Vars{}
|
2018-12-02 18:17:32 +02:00
|
|
|
}
|
2019-01-02 17:20:12 +02:00
|
|
|
if t1.Env == nil {
|
2020-03-29 21:54:59 +02:00
|
|
|
t1.Env = &Vars{}
|
2019-01-02 17:20:12 +02:00
|
|
|
}
|
2020-03-29 21:54:59 +02:00
|
|
|
t1.Vars.Merge(t2.Vars)
|
|
|
|
t1.Env.Merge(t2.Env)
|
2019-01-02 17:20:12 +02:00
|
|
|
|
2018-12-02 18:17:32 +02:00
|
|
|
if t1.Tasks == nil {
|
|
|
|
t1.Tasks = make(Tasks)
|
|
|
|
}
|
2018-07-22 22:54:44 +02:00
|
|
|
for k, v := range t2.Tasks {
|
2022-10-02 07:45:27 +02:00
|
|
|
// We do a deep copy of the task struct here to ensure that no data can
|
|
|
|
// be changed elsewhere once the taskfile is merged.
|
|
|
|
task := v.DeepCopy()
|
2018-12-09 19:54:58 +02:00
|
|
|
|
2022-10-02 07:07:58 +02:00
|
|
|
// Set the task to internal if EITHER the included task or the included
|
|
|
|
// taskfile are marked as internal
|
|
|
|
task.Internal = task.Internal || includedTaskfile.Internal
|
2022-07-22 04:16:14 +02:00
|
|
|
|
2022-10-02 07:07:58 +02:00
|
|
|
// Add namespaces to dependencies, commands and aliases
|
|
|
|
for _, dep := range task.Deps {
|
2018-12-09 19:54:58 +02:00
|
|
|
dep.Task = taskNameWithNamespace(dep.Task, namespaces...)
|
|
|
|
}
|
2022-10-02 07:07:58 +02:00
|
|
|
for _, cmd := range task.Cmds {
|
2021-11-26 11:20:05 +02:00
|
|
|
if cmd != nil && cmd.Task != "" {
|
2018-12-09 19:54:58 +02:00
|
|
|
cmd.Task = taskNameWithNamespace(cmd.Task, namespaces...)
|
|
|
|
}
|
|
|
|
}
|
2022-10-02 07:07:58 +02:00
|
|
|
for i, alias := range task.Aliases {
|
|
|
|
task.Aliases[i] = taskNameWithNamespace(alias, namespaces...)
|
2022-10-02 00:39:44 +02:00
|
|
|
}
|
2022-10-02 07:07:58 +02:00
|
|
|
// Add namespace aliases
|
|
|
|
if includedTaskfile != nil {
|
|
|
|
for _, namespaceAlias := range includedTaskfile.Aliases {
|
2022-10-15 00:27:20 +02:00
|
|
|
task.Aliases = append(task.Aliases, taskNameWithNamespace(task.Task, namespaceAlias))
|
2022-10-02 07:45:27 +02:00
|
|
|
for _, alias := range v.Aliases {
|
|
|
|
task.Aliases = append(task.Aliases, taskNameWithNamespace(alias, namespaceAlias))
|
2022-10-02 07:07:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the task to the merged taskfile
|
|
|
|
t1.Tasks[taskNameWithNamespace(k, namespaces...)] = task
|
2018-07-22 22:54:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2018-09-10 03:29:29 +02:00
|
|
|
|
|
|
|
func taskNameWithNamespace(taskName string, namespaces ...string) string {
|
2019-02-03 01:12:57 +02:00
|
|
|
if strings.HasPrefix(taskName, ":") {
|
|
|
|
return strings.TrimPrefix(taskName, ":")
|
|
|
|
}
|
2018-09-10 03:29:29 +02:00
|
|
|
return strings.Join(append(namespaces, taskName), NamespaceSeparator)
|
|
|
|
}
|