在地图内将模块名称存储在数据库中

时间:2019-05-31 06:50:28

标签: database elixir ecto

我正在使用Elixir / Ecto / Phoenix为IOT应用程序创建通知服务。 此应用中的传感器可以具有多个值,例如

temperature: ["temp"],
accelerometer: ["ax", "ay", "az"]

要创建通用通知策略,我必须将模块名称与传感器一起存储,该传感器将实现处理资格规则的逻辑。而且,由于每个传感器可以具有与其关联的多个值accelerometer -> ax, ay, az,因此模块的不同值可以不同。

  schema("acqdat_sensor_notifications") do
    field(:rule_values, :map)
    field(:alarm_status, :boolean, default: false)
    belongs_to(:sensor, Sensor, on_replace: :delete)

    timestamps()
  end

规则值列将以以下格式存储数据,例如用于加速度传感器。

%{
  "ax" => %{ 
    module: "NotificationPolicyA", 
    module_preferences: {"prefA": valuea, "prefb": valueb}
  },
  "ay" => %{ 
    module: "NotificationPolicyB", 
    module_preferences: {"pref1": value1, "pref2": value2}
  },
}

这里的问题是存储将处理逻辑的模块名称。我不能使用像ecto_enum这样的枚举库,因为这些值存储在一个不是常量的jsonb映射中(在这里不能使用嵌入)。它会随着传感器名称和用于处理它的模块而不断变化。

目前,我正在使用地图进行处理

%{
  NotificationPolicyA: 0, 
  NotificationPolicyA: 1 
  .
  .
  .
}

然后将整数存储在db中,以便在修改模块名称时仅更改映射。

"ax" => %{ 
    module: 0, 
    module_preferences: {"prefA": valuea, "prefb": valueb}
  }

用ecto处理这种情况的替代方法是什么?

1 个答案:

答案 0 :(得分:1)

您可以将模块名称作为字符串存储在数据库中,使用String.to_existing_atom/1进行转换,并使用Kernel.apply/3运行所需的任何功能。

示例:

iex(2)> apply(String.to_existing_atom("Elixir.MyApp.Version"), :run, [1])
0.29.1