Support rail roading with broadcast!/2
davejlong opened this issue · 6 comments
It would be useful if broadcast!/2
supported rail roading. In other words, the data to broadcast should be able to be passed as the first argument of the method. Often times, I find myself writing jobs like:
job :forecast, every: {10, :minutes} do
forecast = APIs.ForecastIO.get location
|> APIs.ForecastIO.forecast_for_widget
broadcast! :forecast, forecast
end
It would be nice syntactic sugar to support:
job :forecast, every: {10, :minutes} do
APIs.ForecastIO.get location
|> APIs.ForecastIO.forecast_for_widget
|> broadcast! :forecast
end
Hey @davejlong @zorbash
I've been using kitto and you're right, I think this feature is helpful, I'm +1 on this.
I think that your approach of adding this as part of #22 might be a good idea, I've made a few tests and I came up with this:
1.- In case that passing a topic name different than the job name is not supported
block = Macro.prewalk (options[:do] || contents[:do]), fn
{:broadcast!, meta, args = [_]} -> {:broadcast!, meta, [name] ++ args}
{:|>, _, [left, {:broadcast!, meta, _}]} -> {:broadcast!, meta, [name] ++ [left]}
ast_node -> ast_node
end
Usage
job :forecast, every: {10, :minutes} do
APIs.ForecastIO.get location
|> APIs.ForecastIO.forecast_for_widget
|> broadcast!
end
2.- In case that passing a topic name different than the job name is supported
block = Macro.prewalk (options[:do] || contents[:do]), fn
{:broadcast!, meta, args = [_]} -> {:broadcast!, meta, [name] ++ args}
{:|>, _, [left, {:broadcast!, meta, [topic]}]} when is_atom(topic) -> {:broadcast!, meta, [topic] ++ [left]}
{:|>, _, [left, {:broadcast!, meta, _}]} -> {:broadcast!, meta, [name] ++ [left]}
ast_node -> ast_node
end
Usage
job :forecast, every: {10, :minutes} do
APIs.ForecastIO.get location
|> APIs.ForecastIO.forecast_for_widget
|> broadcast! :other #or just broadcast! as in approach 1
end
I provided both solutions because at the moment I was looking at this I was a little unsure, but I think that approach 2 will be the correct one.
Let me know your thoughts on this, if you think it's the way to go I'll open a PR.
If I read this right with your suggested changes @saulecabrera we'll support all of the below ways to broadcast:
job :forecast, every: {10, :minutes} do
broadcast! Weather.in(:london)
broadcast! :london_weather, Weather.in(:london)
Weather.in(:london) |> broadcast!
Weather.in(:london) |> broadcast! :uk_weather
end
I appreciate the flexibility of the above, but it kinda breaks the function's contract. I've never seen an Elixir function behave differently when an argument is provided with a pipe. Perhaps we can make a special exception though for the sake of usability. An alternative would be to make a backwards incompatible change and expect broadcast!(atom(), map())
to become broadcast!(map(), atom())
@saulecabrera I think it's better to change lib/kitto/notifier.ex
to support the alternative order of parameters like:
def broadcast!(data, topic) when is_atom(topic), do broadcast!(topic, data)
and then we'll have to Macro.prewalk
only for:
Weather.in(:london) |> broadcast!
and it'll still be possible to call broadcast! Weather.in(:london), :london_weather
.
Will you create a PR for this?
Hi @zorbash
I've never seen an Elixir function behave differently when an argument is provided with a pipe.
Makes perfect sense. From the previous discussion it was not clear for me which approach will be the best one since this the change can be made in both ways. Your arguments make it clear now.
Will you create a PR for this?
Yes I'll submit a pull request for this.