-
-
Notifications
You must be signed in to change notification settings - Fork 640
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Ability to loop over values #1220
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Awesome work as always! 👏 👏 👏
The implementation seems great!
I added a couple of comments for discussion.
taskfile/cmd.go
Outdated
Task string | ||
ForEach *ForEach |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I wonder if we should call it only for
. It'd would align well with the Go syntax. I don't have a strong opinion, though. foreach
is also fine.
If we keep foreach
, I'd add a struct tag here because otherwise I believe it'd also be possible to declare it as for_each
, leading to inconsistencies.
ForEach *ForEach | |
ForEach *ForEach `yaml:"foreach"` |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Happy to use for
instead. It keeps things nice and concise. foreach
was based off the comments in #82.
Edit - Updated
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@andreynering Quick question on this. foreach: source
kind of made sense because it reads well in English.
- "Do thing for each source"
Now that we have for
instead of foreach
, I think it makes more sense to use the attribute name directly rather than the singular form of the noun. i.e. for: sources
reads better to me than for: source
.
This would also make it easier to be consistent if we ever loop over other fields. for: generates
makes more sense than trying to work out what the singular form of generates
is.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
^^^ I have updated to for: sources
as mentioned.
} | ||
// Get the list from the task sources | ||
if cmd.ForEach.From == "source" { | ||
list, err = fingerprint.Globs(new.Dir, new.Sources) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I noticed that foreach: source
is returning absolute paths like:
/Users/andrey/Developer/andrey/task/args/args.go
instead relative like
args/args.go
We need to discuss which would be more appropriate to the user: absolute or relative paths. Ideally, whichever option we choose, the user should be able to convert one into another. I'm inclined to prefer relative. We can use filepath.Rel
to convert an abs path to relative.
What may become tricky here is that we should take into account the task directory and not the user directory when converting.
Maybe we should add a TASK_DIR
special var similar to the other special variables we already have. Then, the user would be able to do something like echo {{join .TASK_DIR .ITEM}}
to get the absolute path.
(This join
function doesn't seem to be included into slim-sprig nor on the list of functions we expose ourselves to the user. We will need to add it. Given a rel
function is also missing, we could probably add it as well.)
Any thoughts?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a super interesting point that I didn't consider. I completely agree that having functions to switch between relative (rel
) and absolute (abs
) URLs would be really useful (and also a join
).
However, I'm not at all sure which type of path would be most useful to the user. Out of curiosity, is there any reason that you lean towards relative over absolute?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If feels "cleaner", I guess. Less verbose.
There's no right on wrong on this decision, though.
I think I'm fine if we keep as absolute as well, but with the proper function to convert + an example on the docs on how to do it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I took a look at this today. I think adding abs
and rel
functions to the templates is going to be really hard. Some variation of join
is easy since it always takes the same arguments. However, as you eluded to, abs
and rel
would need to be aware of the context (task directory).
There isn't really a simple way of doing this. You could make a function that accepts the current directory as a parameter, but that is not very user friendly IMO. If we wanted to make the function itself context aware (so that a user doesn't have to pass in the current directory), we'd need to create a set of templateFuncs
for each call to Replace()
to avoid race conditions. This is a drastic change from the current contextless functions defined in the init()
function.
The other solution would be to add something like pathJoin
to sprig (join
already exists for slices) and the TASK_DIR
variable you mentioned.
I prefer the second option, but my gut feeling is to leave these out of this PR and add them later if users request it. At least this will mean the main functionality is available.
As suggested, I have changed the paths created by sources
to be relative as I have no strong opinion on which format they should use. I think most users would write them relatively if they were to manually list them out, so this makes sense to me for now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You could make a function that accepts the current directory as a parameter.
I think this route would be an appropriate middle ground. What do you think?
I just looked at the Go's documentation and realized that filepath.Rel
actually already asks for the base path, so asking on our side would be consistent.
Regarding abs
, it would actually just be a pathJoin(<working-dir>, <path>)
, so maybe we could just skip abs
to something we would consider to do in the future.
The important here is to properly document this in the documentation so it's easy enough for users to find and use.
Last thought, maybe we should add path
to all functions for clarity:
joinPath
relPath
absPath
(future)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have added the joinPath
and relPath
functions. I wasn't sure if these should be in slim-sprig or Task, so I've just added them to Task for now.
I did also look into adding a TASK_DIR
special variable, but this is a bit awkward at the moment as t.Dir
is not resolved until after the variable compilation happens and t.Dir
also accepts variables which need to be resolved. There are ways to solve this, but I think doing lazy variables (in some form) will solve that issue anyway, so maybe this is another future thing.
I've also added some docs to the for
section describing how you would go about converting the relative paths to absolute ones.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I have added the
joinPath
andrelPath
functions. I wasn't sure if these should be in slim-sprig or Task, so I've just added them to Task for now.
That's fine. We can always move them to slim-sprig
later if we want to.
07bc055
to
1fb5c60
Compare
I love the addition but would want I would also use it on the command level, but for different purposes, so i welcome the addition. Sort of inside smaller more focused task targets which end up being just one command (e.g. "build" and "push"), so it also seems that being one level up would behave similarly. My wish would be we could reference vars for reuse of the []; but it doesn't seem like much of a problem to address, yaml anchors exist which would basically enable that. |
I don't think this is necessary. If your task is a single command then obviously, the current syntax is no problem. You just loop over your single command. However (correct me if I'm wrong), I'm assuming you want this for cases when you have multiple commands that you want to loop over. For example: build-and-push-all:
for: [ 'foo', 'bar' ]
cmds:
- docker build -t example:{{.ITEM}} ./{{.ITEM}}
- docker push ... In these cases, I think the better way is to create a distinct task that will run the commands for a single iteration, and then call that from somewhere else: build-and-push-all:
cmds:
- for: [ 'foo', 'bar' ]
task: build-and-push
vars:
ITEM: {{.ITEM}}
build-and-push:
cmds:
- docker build -t example:{{.ITEM}} ./{{.ITEM}}
- docker push ... This is more verbose, but it has the advantage that the Happy to hear other opinions or if you can provide an example where this approach doesn't work as well.
I think
I'm not quite sure I fully understand. Could you elaborate? |
This example would be more like:
So, the
But, as we're here, this could be expressed as:
Just having that The ITEM concerns are already addressed with |
About the yaml anchors, i forget the syntax, but essentially:
Consider it more pseudo code, I don't think I remember the syntax fully, nor if task yaml parser supports it. Sometimes it's a bit of a bother carrying values over vars and i think yaml anchors could be usable. Something about having default values for vars here would be also possible with anchors rather than /offtopic |
I have done looping with recursion before (see this). In cases where I used it, I had both the task name and some additional parameters in each item, and that was obtained from a variable that was dynamically populated. I think it may be possible with this construct using ndjson and a newline as the delimiter, though I would just like to confirm that it will be. So basically, I would want to be able to do something like this: version: '3'
tasks:
default:
vars:
my_var: |
{ "task": "foo", "args": { "x": 1 } }
{ "task": "foo", "args": { "x": 1 } }
cmds:
- for:
var: my_var
split: "\n"
task: "{{ (.ITEM | mustFromJson).task }}"
vars:
ITEM_ARGS: '{{(.ITEM | mustFromJson).args | mustToJson}}'
foo: {} |
I installed this branch locally and took it for a spin, and I was happy with it. Any chance we'll see it in an imminent release? |
This is very close, just require small adjustments to abs/rel path handling. I see this being available in the next release within a few weeks. 🙂 |
@kevinmichaelchen @andreynering looking to get the outstanding comments fixed in the next couple of days. Watch this space 👍 |
@aucampia For some reason (I haven't looked into it) |
| --------- | -------- | ---------------- | -------------------------------------------- | | ||
| `var` | `string` | | The name of the variable to use as an input. | | ||
| `split` | `string` | (any whitespace) | What string the variable should be split on. | | ||
| `as` | `string` | `ITEM` | The name of the iterator variable. | |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nice.
I've missed the ability of |
@pd93
I would expect this to be possible:
Or some sort of quoted literal value, but i suppose this ends up being a |
A smaller caveat:
could the var passing here be explicit, dropping the last 2 lines? |
@titpetric If you want to loop over literal values then you either need to declare these values in sync-not-ok:
vars:
MY_VAR: "tyk tyk-analytics tyk-analytics-ui tyk-docs exp"
cmds:
- for:
var: MY_VAR
as: PROJECT
cmd: echo {{.project}} (splitting on whitespace is the default), but you can specify a different character if you want with OR, you can simply define the list as an array on the sync-not-ok:
cmds:
- for: [tyk tyk-analytics tyk-analytics-ui tyk-docs exp]
cmd: echo {{.ITEM}} Note that the second choice doesn't currently support the ability to use |
yeah, i was thinking |
Fixes #82
This PR adds the ability to loop over values in Task. This is useful when you want to run a command multiple times with different values. It supports looping over static lists, variables, and task sources and can be used with commands and tasks.
I've implemented this slightly differently to the suggestions in #82 as I believe that implementing this at the
cmds
level gives us a lot more flexibility. It means we can run other commands before/after doing loops, but still allows us to loop over commands and tasks and pass around variables.I've pasted the docs added in this PR below as they provide a good idea of how to use the new functionality:
Looping over values
Task allows you to loop over certain values and execute a command for each.
There are a number of ways to do this depending on the type of value you want to
loop over.
Looping over a static list
The simplest kind of loop is an explicit one. This is useful when you want to
loop over a set of values that are known ahead of time.
Looping over your task's sources
You are also able to loop over the sources of your task:
This will also work if you use globbing syntax in your sources. For example, if
you specify a source for
*.txt
, the loop will iterate over all files thatmatch that glob.
Looping over variables
To loop over the contents of a variable, you simply need to specify the variable
you want to loop over. By default, variables will be split on any whitespace
characters.
If you need to split on a different character, you can do this by specifying the
split
property:All of this also works with dynamic variables!
Renaming variables
If you want to rename the iterator variable to make it clearer what the value
contains, you can do so by specifying the
as
property:Looping over tasks
Because the
for
property is defined at thecmds
level, you can also use italongside the
task
keyword to run tasks multiple times with differentvariables.
Or if you want to run different tasks depending on the value of the loop: