ERLANG с JSON - PullRequest
       32

ERLANG с JSON

0 голосов
/ 25 января 2019

Я запускаю следующую команду в erlang,

os:cmd("curl -k -X GET http://10.210.12.154:10065/iot/get/task").

Это дает вывод в формате JSON, как это,

{"data":[
    {"id":1,"task":"Turn on the bulb when the temperature in greater than 28","working_condition":1,"depending_value":"Temperature","action":"123"},
    {"id":2,"task":"Trun on the second bulb when the temperature is greater than 30","working_condition":0,"depending_value":"Temperature","action":"124"}
]}

Я хочу классифицировать эти данные по Id, task, зависящее_значение, действие.Это как положить их на стол.Я хочу легко найти то, что является зависимым значением, рабочим условием и действием для Id = 1.Как я могу это сделать?

1 Ответ

0 голосов
/ 26 января 2019

Это дает вывод в формате JSON, как этот.

{"data":[{"id":1,"t ...

Очень сомнительно. В документах сказано, что os: cmd () возвращает строку , которая не начинается с {. Также обратите внимание, что строка не является даже типом данных erlang, скорее двойные кавычки являются ярлыком для создания list of integers, и список целых чисел не очень полезен в вашем случае.

Вот два варианта:

  1. Вызовите list_to_binary() в списке целых чисел, возвращаемых os:cmd(), чтобы преобразовать в binary.

  2. Вместо os:cmd() используйте HTTP-клиент erlang, например hackney , который вернет json как binary.

Причина, по которой вам нужен двоичный файл, заключается в том, что тогда вы можете использовать модуль erlang json, например jsx , для преобразования двоичного файла в карту erlang (что может быть тем, что вам нужно?).

Вот как это будет выглядеть:

3> Json = <<"{\"data\": [{\"x\": 1, \"y\": 2}, {\"a\": 3, \"b\": 4}] }">>. 
<<"{\"data\": [{\"x\": 1, \"y\": 2}, {\"a\": 3, \"b\": 4}] }">>

4> Map = jsx:decode(Json, [return_maps]).
#{<<"data">> =>
      [#{<<"x">> => 1,<<"y">> => 2},#{<<"a">> => 3,<<"b">> => 4}]}

5> Data = maps:get(<<"data">>, Map).     
[#{<<"x">> => 1,<<"y">> => 2},#{<<"a">> => 3,<<"b">> => 4}]

6> InnerMap1 = hd(Data).   
#{<<"x">> => 1,<<"y">> => 2}

7> maps:get(<<"x">>, InnerMap1).
1

... положить их на стол. Я хочу легко найти то, что является в зависимости от значения, условий работы и действия для Id = 1.

Erlang имеет различные реализации таблиц: ets, dets и mnesia. Вот пример ets:

-module(my).
-compile(export_all).

get_tasks() ->
    Method = get,

    %See description of this awesome website below.
    URL = <<"https://my-json-server.typicode.com/7stud/json_server/db">>,

    Headers = [],
    Payload = <<>>,
    Options = [],

    {ok, 200, _RespHeaders, ClientRef} =
        hackney:request(Method, URL, Headers, Payload, Options),
    {ok, Body} = hackney:body(ClientRef),
    %{ok, Body} = file:read_file('json/json.txt'),  %Or, for testing you can paste the json in a file (without the outer quotes), and read_file() will return a binary.

    Map = jsx:decode(Body, [return_maps]),
    _Tasks = maps:get(<<"data">>, Map).

create_table(TableName, Tuples) ->
    ets:new(TableName, [set, named_table]),
    insert(TableName, Tuples).

insert(_Table, []) ->
    ok;
insert(Table, [Tuple|Tuples]) ->
    #{<<"id">> := Id} = Tuple,
    ets:insert(Table, {Id, Tuple}),
    insert(Table, Tuples).

retrieve_task(TableName, Id) ->
    [{_Id, Task}] = ets:lookup(TableName, Id), 
    Task.

По умолчанию таблица типа ets set гарантирует, что первая позиция во вставленном кортеже - это уникальный ключ (или вы можете явно указать другую позицию в кортеже в качестве уникального ключа).

** Если у вас есть учетная запись на github, я обнаружил действительно классный веб-сайт, который позволяет вам помещать файл json в новый репозиторий на github, и этот веб-сайт будет обслуживать этот файл как json. Проверьте это на https://my -json-server.typicode.com :

Как

  1. Создать репозиторий на GitHub (<your-username>/<your-repo>)
  2. Создайте файл db.json [в хранилище].
  3. Посетите https://my -json-server.typicode.com / <your-username>/<your-repo>, чтобы получить доступ к вашему серверу

Вы можете увидеть URL, который я использую в коде, который можно получить, нажав на ссылку на странице предоставленного сервера и скопировав URL в адресную строку вашего браузера.

В оболочке:

.../myapp$ rebar3 shell
===> Verifying dependencies...
===> Compiling myapp
src/my.erl:2: Warning: export_all flag enabled - all functions will be exported

Erlang/OTP 20 [erts-9.3] [source] [64-bit] [smp:4:4] [ds:4:4:10] [async-threads:1] [hipe] [kernel-poll:false]
Eshell V9.3  (abort with ^G)

1> ===> The rebar3 shell is a development tool; to deploy applications in production, consider using releases (http://www.rebar3.org/docs/releases) 
===> Booted unicode_util_compat
===> Booted idna
===> Booted mimerl
===> Booted certifi
===> Booted ssl_verify_fun
===> Booted metrics
===> Booted hackney

1> Tasks = my:get_tasks().     
[#{<<"action">> => <<"123">>,
   <<"depending_value">> => <<"Temperature">>,<<"id">> => 1,
   <<"task">> =>
       <<"Turn on the bulb when the temperature in greater than 28">>,
   <<"working_condition">> => 1},
 #{<<"action">> => <<"124">>,
   <<"depending_value">> => <<"Temperature">>,<<"id">> => 2,
   <<"task">> =>
       <<"Trun on the second bulb when the temperature is greater than 30">>,
   <<"working_condition">> => 0}]

2> my:create_table(tasks, Tasks).
ok

3> my:retrieve_task(tasks, 1).   
#{<<"action">> => <<"123">>,
  <<"depending_value">> => <<"Temperature">>,<<"id">> => 1,
  <<"task">> =>
      <<"Turn on the bulb when the temperature in greater than 28">>,
  <<"working_condition">> => 1}

4> my:retrieve_task(tasks, 2).   
#{<<"action">> => <<"124">>,
  <<"depending_value">> => <<"Temperature">>,<<"id">> => 2,
  <<"task">> =>
      <<"Trun on the second bulb when the temperature is greater than 30">>,
  <<"working_condition">> => 0}

5> my:retrieve_task(tasks, 3).
** exception error: no match of right hand side value []
     in function  my:retrieve_task/2 (/Users/7stud/erlang_programs/old/myapp/src/my.erl, line 58)

6> 

Обратите внимание, что идентификатор находится справа в конце одной из строк. Кроме того, если вы обнаружите какие-либо ошибки в оболочке, оболочка автоматически перезапустит новый процесс, и таблица ets будет уничтожена, поэтому вам придется создать ее заново.

rebar.config:

{erl_opts, [debug_info]}.
{deps, [
    {jsx, "2.8.0"},
    {hackney, ".*", {git, "git://github.com/benoitc/hackney.git", {branch, "master"}}}
]}.
{shell, [{apps, [hackney]}]}. % This causes the shell to automatically start the listed apps.  See https://stackoverflow.com/questions/40211752/how-to-get-an-erlang-app-to-run-at-starting-rebar3/45361175#comment95565011_45361175

ЦСИ / myapp.app.src:

{application, 'myapp',
 [{description, "An OTP application"},
  {vsn, "0.1.0"},
  {registered, []},
  {mod, {'myapp_app', []}},
  {applications,
   [kernel,
    stdlib
   ]},
  {env,[]},
  {modules, []},

  {contributors, []},
  {licenses, []},
  {links, []}
 ]}.

Но, согласно документациям о зависимостях rebar3 :

Вы должны добавить каждую зависимость в ваше приложение или файлы app.src:

Итак, я думаю, src/myapp.app.src должно выглядеть так:

{application, 'myapp',
 [{description, "An OTP application"},
  {vsn, "0.1.0"},
  {registered, []},
  {mod, {'myapp_app', []}},
  {applications,
   [kernel,
    stdlib,
    jsx,
    hackney
   ]},
  {env,[]},
  {modules, []},

  {contributors, []},
  {licenses, []},
  {links, []}
 ]}.
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...