JSON将geo库中的GEO.Point编码为人类可读形式

时间:2017-06-01 06:48:15

标签: elixir ecto geo elixir-framework elixir-poison

我的这个架构有一个Geo Geo.Point

defmodule Api.Shop do
  use Ecto.Schema
  import Ecto.Changeset
  import Api.Repo
  import Ecto.Query

  @derive {Poison.Encoder, only: [:name, :place_id, :geo_json, :distance]}
  schema "shops" do
    field :name, :string
    field :place_id, :string
    field :point, Geo.Point
    field :geo_json, :string, virtual: true
    field :distance, :float, virtual: true

    timestamps()
  end

  def encode_model(shop) do
    %Api.Shop{shop | geo_json: Geo.JSON.encode(shop.point) }
  end

  defimpl Poison.Encoder, for: Api.Shop do
    def encode(shop, options) do
      shop = Api.Shop.encode_model(shop)
      Poison.Encoder.Map.encode(Map.take(shop, [:id, :name, :geo_json]), options)
    end
  end

  def changeset(shop, params \\ %{}) do
    shop
    |> cast(params, [:name, :place_id, :point])
    |> validate_required([:name, :place_id, :point])
    |> unique_constraint(:place_id)
  end......
end

当我在查询中返回shop.point字段时:

  def create_query_no_keyword(categories, shop_ids) do
    products_shops_categories = from p in Product,
    distinct: p.id,
    join: ps in ProductShop, on: p.id == ps.p_id,
    join: s in Shop, on: s.id == ps.s_id,
    join: pc in ProductCategory, on: p.id == pc.p_id,
    join: c in Subcategory, on: c.id == pc.c_id,
    where: c.id in ^categories,
    where: s.id in ^shop_ids,
    group_by: [p.id, p.name, p.brand],
    select: %{product: p, categories: fragment("json_agg( DISTINCT (?, ?)) AS category", c.id, c.name), shops: fragment("json_agg( DISTINCT (?, ?, ?)) AS shop", s.id, s.name, s.point)}
  end

返回的内容实际上是0101000020E6100000A3BDB0EB0DD9654030AC2C1BE76D42C0格式错误 - WKB。我希望编码为具有可读坐标的WKT。

如何使s.point成为WKT格式,从而在查询返回时具有坐标?

1 个答案:

答案 0 :(得分:1)

我发现this Stack Exchange GIS answer是解决方案:

  

将此用于点对象:

SELECT ST_AsText(the_geom) 
       FROM myTable; and viewing X,Y and geom object:

SELECT ST_X(the_geom), ST_Y(the_geom), ST_AsText(the_geom) 
       FROM myTable;

Geo库正在使用PostGIS,解决方案是PostGIS特定的。您需要使用ST_AsText或PostGIS中的ST_XST_Y选择列。

我的选择语句改为:

select: %{product: p, categories: fragment("json_agg( DISTINCT (?, ?)) AS category", c.id, c.name), shops: fragment("json_agg( DISTINCT (?, ?, ST_X(?), ST_Y(?))) AS shop", s.id, s.name, s.point, s.point)}