simotin13's message

simotin13といいます。記事の内容でご質問やご意見がありましたらお気軽にコメントしてください\^o^/

Elixir入門 ~8日目 HTTPoisonを動かすにはElixir1.2が必要だった~

昨日の続き。

mixが動くようになって、 mix.exs を編集して mix deps.get でモジュールをダウンロードするところまではできた。

と、思っていた。


今日はGithub

github.com


を参考にいざ動かそうとしたら以下のようなエラーメッセージがでた・・・

今度はなんだ?いい加減エラーメッセージでコンソールが真っ赤に染まるのには飽きてきた。

$ iex -S mix
Erlang/OTP 18 [erts-7.3] [source] [64-bit] [async-threads:10] [kernel-poll:false]

==> httpoison
warning: the dependency httpoison requires Elixir "~> 1.2" but you are running on v1.1.0-dev

== Compilation error on file lib/httpoison/base.ex ==
** (CompileError) lib/httpoison/base.ex:452: function '<-'/2 undefined
    (stdlib) lists.erl:1337: :lists.foreach/2
    (stdlib) erl_eval.erl:670: :erl_eval.do_apply/6

「httpoison を動かしたかったらElixir 1.2 を入れな!」

ということらしい。

httpoison のバージョンとかおとしたらもしかしたら動くのかもしれないが今日はやめておく。

その前に一言。

やりたいことができないこんなElixirじゃ、ぽいぞん。

そもそもどのレイヤーで依存関係が発生しているのかよく分からないがそんなに動かなくなるものかね・・・

mix 動かしたいだけなのに。


まぁ愚痴ってもしょうがないので今度はElixirのバージョンアップから試そう。

Elixir入門 ~7日目 mixが動かない原因が分かった~

mix が動かない原因が分かった。

「elixir iex INFO REPORT」

ググるといくつか情報が出ていて、日本語の情報だとQiitaのこちらの記事が参考になりそうです。

qiita.com
※参考にさせて頂き、ありがとうございます (m_m)

手元で確認したバージョンと結果

ということで、原因は要するに、 Erlang/OTPとElixirのバージョンの相性ということで間違いなさそう。

Linux環境では基本的にapt-getでインストールしていたけど、インストールされるOTPのバージョンが違うために偶然にもうまく動くパターンもあった。

多分apt使うときにupdateとか叩き忘れているという・・・

手元で確認したバージョンとかの組み合わせを一応あげておきます。

OS Elixir Erlang/OTP 動作 インストール方法
Linux Mint Elixir 1.1 18 OK apt
Linux Mint Elixir 1.1 20 NG apt
Windows10 Elixir 1.1 20 OK インストーラ

試したかったこ

さて、とりあえず手元にうごく環境もあるということでmixの勉強をしてみる。

Mixの使い方を知るには、

elixirschool.com

がよさそう。

早速 mix new で新しいプロジェクトを作る。
やりたいのはHTTP通信(クライアント側)HTTPクライアントの機能を使ってみたいのでプロジェクト名はhttpとした。
HTTPPoisonというモジュールがあるらしい。

$ mix new http                                                                      
* creating README.md
* creating .gitignore
* creating mix.exs
* creating config
* creating config/config.exs
* creating lib
* creating lib/http.ex
* creating test
* creating test/test_helper.exs
* creating test/http_test.exs

Your mix project was created successfully.
You can use mix to compile it, test it, and more:

    cd http
    mix test

Run `mix help` for more commands.

mix.exsの編集

mix.exs がRubyでいうところのGemfileに相当するファイルのようだ。

github.com

qiita.com

GithubやQiitaの記事などを参考にさせて頂き適当に編集。

mix deps.get が失敗した

$ mix deps.get

が bundle install 相当のコマンドになるようだ。

最初にmix deps.getを実行したときにモジュールの取得に失敗していた。

20:10:50.392 [error] SSL: :certify: ssl_handshake.erl:1474:Fatal error: certificate expired

Failed to check for new Hex version
Failed to fetch record for 'hexpm/httpoison' from registry (using cache)

20:10:50.393 [error] SSL: :certify: ssl_handshake.erl:1474:Fatal error: certificate expired


20:10:50.394 [error] SSL: :certify: ssl_handshake.erl:1474:Fatal error: certificate expired

ログをみてみるとSSL認証で失敗しているらしい。Elixirを試しているのはVM Ware Player上のLinux Mintだが案の定時間が大幅にずれていた。
時刻を合わせて再度試したらうまくいったようだ。いろいろとトラブルが発生するものだ・・・


続きは明日にしよう。

Elixir入門 ~6日目 elixirのインストールに失敗してた~

今日はmixを使っていろいろ試してみようと思ったらmix コマンドがちゃんと動かなかった。

どうもelixir/erlangのインストールに失敗してたっぽい。

$ mix

=INFO REPORT==== 10-Sep-2017::17:13:16 ===
    application: logger
    exited: {{shutdown,
                 {failed_to_start_child,'Elixir.GenEvent',
                     {undef,
                         [{gen,debug_options,[[]],[]},
                          {'Elixir.GenEvent',init_it,6,
                              [{file,"lib/gen_event.ex"},{line,538}]},
                          {proc_lib,init_p_do_apply,3,
                              [{file,"proc_lib.erl"},{line,247}]}]}}},
             {'Elixir.Logger.App',start,[normal,[]]}}
    type: temporary
** (Mix) The task app.start could not be found

インストールしたけど使っていなかったwindows版だとこんなエラーメッセージでてない・・・

C:\>mix
** (Mix) "mix" with no arguments must be executed in a directory with a mix.exs file

Usage: mix [task]

Examples:

    mix             - Invokes the default task (current: "mix run")
    mix new PATH    - Creates a new Elixir project at the given path
    mix help        - Lists all available tasks
    mix help TASK   - Prints documentation for a given task

C:\>mix -v
Erlang/OTP 20 [erts-9.0] [64-bit] [smp:8:8] [ds:8:8:10] [async-threads:10]

Mix 1.5.1


elixir入門者過ぎてこのエラーメッセージの意味がわからないけどなんか必要なライブラリとか入っていないような感じだろうか。

明日ちゃんと調べてみよう。

Elixir入門 ~5日目 ファイルの読み書き~

今日は帰りが遅くなったのであまり時間が取れない。

ファイルの読みこみだけ試してみた。

CSV形式のファイルを読んでみる。

sample.csv とする。

aaa,1
bbb,2
ccc,3

ファイルアクセスはFileモジュールを使うらしい。
読み込みはreadかopen

iex(1)> File.read("sample.csv")
{:ok, "aaa, 1\nbbb, 2\nccc, 3\n"}

結果はタプルで返ってくる。

最初の

:ok

はアトム型といってrubyのシンボル相当らしい。

タプルで結果が返ってくるのは違和感があるが、読み込み結果を何らかの形で返す必要があるのでこういう仕様なんだろう。

存在しないファイルを指定すると

{:error, :enoent}

という結果が返ってくる。

カンマ区切りの分割にはStringのsplit関数が使えるらしい。
結果はリストになる。

iex(1)> String.split("aaa","bbb","ccc", ",")
["aaa", "bbb", "ccc"]

複数行・複数カラムの扱いはEnumモジュールのeachとかを使うとrubyみたいな感じでかけるらしい。

今日は疲れてるので早く寝よう、また明日だ。

Elixir入門 ~4日目 モジュール・名前付き関数~

昨日は無名関数について簡単に勉強したので今日は名前付き関数について勉強してみる。

モジュール・関数については、は参考書籍として使っているプログラミングElixirの6章で説明があるので読みながら勉強してみる。

プログラミングElixir

プログラミングElixir

モジュールと名前付き関数

名前付き関数はモジュールの内側で書く必要があるそうだ。

6章のtimes.exsを写経して動かしてみた。

defmodule Times do
  def double(n) do
    n * 2
  end 
end

モジュールの定義は

defmodule モジュール名 do
end

と書く。do~endで囲むあたりはrubyと似ている。

名前付き関数は、モジュールの内部で、

def 関数名(引数) do
end

とすればよいみたい。
elixirは動的型付けなので引数は引数名だけでよい。

ファイルに書いたコードを実行するためにはいくつか方法がある。

1.iexの起動時に引数でファイル名を指定してロードする方法。

iex times.exs

2.iex内でc ファイル名として、c ヘルパーの機能でロードする方法。


コンパイルコマンドであるelixircを使う方法はここでは紹介されていない。
検索してみても見つからなかったのでこの本自体ではelixircに関する説明がなさそうだ。

名前付き関数の呼び出しは

モジュール名.関数名(引数)

の形になる。

iex(1)> Times.double(3)
6
iex(2)> 

気になって試したこと

  • iex ファイル名を指定して読み込めるのは1ファイルだけのようだ。2ファイル目以降を指定してもロードされていないっぽい。
  • モジュール名は大文字で、関数名は小文字ではじめないといけないみたい。
defmodule times do
  def double(n) do
    n * 2
  end 
end

defmodule Times do
  def Double(n) do
    n * 2
  end 
end

コンパイルに失敗する。コンパイルに失敗するとiexは起動せずエラーメッセージが出力されて終了する。

$ iex times.exs
Erlang/OTP 20 [erts-9.0] [source] [64-bit] [smp:4:4] [ds:4:4:10] [async-threads:10] [hipe] [kernel-poll:false]


=INFO REPORT==== 10-Sep-2017::15:42:48 ===
    application: logger
    exited: {{shutdown,
                 {failed_to_start_child,'Elixir.GenEvent',
                     {undef,
                         [{gen,debug_options,[[]],[]},
                          {'Elixir.GenEvent',init_it,6,
                              [{file,"lib/gen_event.ex"},{line,538}]},
                          {proc_lib,init_p_do_apply,3,
                              [{file,"proc_lib.erl"},{line,247}]}]}}},
             {'Elixir.Logger.App',start,[normal,[]]}}
    type: temporary
** (CompileError) times.exs:1: undefined function times/0
    (elixir) expanding macro: Kernel.defmodule/2
    times.exs:1: (file)

iexの起動時にINFO REPORT==== というのがでているがこれは何なのかよくわかっていない。
別に害があるわけではないけどなんか気になる。

今は意図的に失敗させてわけだけど、このエラーメッセージから間違いに気づくのは慣れないと難しいかもしれないな。

ちなみに先頭以外は大文字でも小文字でもよくて、

defmodule TIMES do
  def dOuble(n) do
    n * 2
  end 
end

は問題なくコンパイルが通る。

  • モジュール外で関数は定義できない。

モジュールの外側で関数を定義しようとするとどうなるか?

def double(n) do
    n * 2
end

コンパイル時のメッセージは以下の通り、

** (ArgumentError) cannot invoke def/2 outside module
    (elixir) lib/kernel.ex:3772: Kernel.assert_module_scope/3
    (elixir) lib/kernel.ex:2977: Kernel.define/4
    (elixir) expanding macro: Kernel.def/2
    test.exs:1: (file)

cannot invoke def/2 outside module と書いているのでこれは比較的わかりやすい気がする。

rubyのようにトップレベルでの関数定義はできないようだ。
もっともrubyの場合はKernelモジュールの省略という話があるので、思想としてはそんなに違わないのかもしれない。

今日はこれくらいにしておこう。

4日目を終えて

少しずつの勉強なので恐ろしく進展が少ない。新しい言語を勉強するときによくやるのが、

  • CSV,JSONフォーマットを扱うファイル入出力
  • ソケットプログラミング
  • HTTPのクライアント(wget的なコード)

なんかをよくやる。

要するに使い捨てのツールを書いてみるということなんだけど、Elixirは関数型言語だしそういったお手軽スクリプティングとして使うのは違う気がする。(別にしたらだめというわけではないだろうが)

File – Elixir v1.5.1

を少し読んでみたけどまた覚えないといけないことがいろいろ出てきそうだ。

明日はファイルアクセスを少し触ってみよう。