如何在Elixir或Phoenix框架中安排代码每隔几个小时运行一次?

时间:2015-08-19 01:47:46

标签: elixir phoenix-framework

所以,让我们说我想发送一堆电子邮件或重新创建站点地图或者每4个小时一次,我怎么能在凤凰城或者只是用Elixir?

7 个答案:

答案 0 :(得分:333)

有一个简单的替代方案,不需要任何外部依赖:

defmodule MyApp.Periodically do
  use GenServer

  def start_link do
    GenServer.start_link(__MODULE__, %{})
  end

  def init(state) do
    schedule_work() # Schedule work to be performed at some point
    {:ok, state}
  end

  def handle_info(:work, state) do
    # Do the work you desire here
    schedule_work() # Reschedule once more
    {:noreply, state}
  end

  defp schedule_work() do
    Process.send_after(self(), :work, 2 * 60 * 60 * 1000) # In 2 hours
  end
end

现在在你的监督树上:

worker(MyApp.Periodically, [])

答案 1 :(得分:27)

Quantum允许您在运行时创建,查找和删除作业。

此外,您可以在创建cronjob时将参数传递给任务函数,如果您对UTC不满意,甚至可以修改时区:

https://github.com/c-rack/quantum-elixir

如果您的应用程序作为多个独立实例(例如Heroku)运行,则Redis支持的作业处理器也支持任务调度:

Exq:https://github.com/akira/exq

Toniq:https://github.com/joakimk/toniq

Verk:https://github.com/edgurgel/verk

答案 2 :(得分:22)

您可以使用erlcron。你像

一样使用它
job = {{:weekly, :thu, {2, :am}},
  {:io, :fwrite, ["It's 2 Thursday morning~n"]}}

:erlcron.cron(job)

job是一个2元素的元组。第一个元素是表示作业计划的元组,第二个元素是函数或MFA(模块,函数,Arity)。在上面的示例中,我们每周四凌晨2点运行:io.fwrite("It's 2 Thursday morning")

希望有所帮助!

答案 3 :(得分:5)

我使用了量子库Quantum- Elixir 请遵循以下说明。

#your_app/mix.exs
defp deps do
  [{:quantum, ">= 1.9.1"},  
  #rest code
end



#your_app/mix.exs
def application do
  [mod: {AppName, []},
   applications: [:quantum,
   #rest code         
 ]]
end

#your_app/config/dev.exs
config :quantum, :your_app, cron: [
  # Every minute
  "* * * * *": fn -> IO.puts("Hello QUANTUM!") end
]

全套。通过运行以下命令启动服务器。

iex -S mix phoenix.server 

答案 4 :(得分:2)

我发现:timer.send_interval/2GenServer(在the accepted answer中使用)相比,与Process.send_after/4结合使用的人机工程学要好得多。

:timer.send_interval/2不必每次都要重新安排您的通知,而是设置了一个不断接收消息的时间间隔-无需像接受的答案那样继续拨打schedule_work()。 / p>

defmodule CountingServer do
  use GenServer

  def init(_) do
    :timer.send_interval(1000, :update)
    {:ok, 1}
  end

  def handle_info(:update, count) do
    IO.puts(count)
    {:noreply, count + 1}
  end
end

每1000毫秒(即每秒一次),将调用IntervalServer.handle_info/2,打印当前的count,并更新GenServer的状态(count + 1),为您提供如下输出:

1
2
3
4
[etc.]

答案 5 :(得分:0)

除了使用Process.send_after之外,您还可以使用:timer.apply_interval

答案 6 :(得分:0)

Quantum很棒,我们在工作中将其用作凤凰前端的cron替代品,并且还在real-time中添加了非常整洁的作业。