Documentação
Documentando código em Elixir.
Anotações
Já comentamos muito sobre o que faz uma documentação ser de qualidade porem isto continua sendo uma controvérsia dentro do mundo da programação. No entanto, nós podemos concordar que documentação é importante tanto para nós, quanto para aqueles que trabalham com a nossa base de código.
Elixir trata de documentação como uma cidadã de primeira classe, oferecendo várias funções de acesso e geração de documentação para seus projetos. O núcleo do Elixir nos fornece muitos atributos diferentes para anotar uma base de código. Vejamos 3 maneiras:
-
#
- Para documentação em linha. -
@moduledoc
- Para documentação em nível de módulo. -
@doc
- Para documentação em nível de função.
Documentação em Linha
Provavelmente, a maneira mais simples de comentar o seu código é com comentários em linha. Semelhante a Ruby ou Python, comentário em linha do Elixir é determinado com um #
, muitas vezes conhecido como um sustenido
, ou um hash dependendo de onde você vive no mundo.
Observe este script em Elixir (greeting.exs):
# Outputs 'Hello, chum.' to the console.
IO.puts("Hello, " <> "chum.")
Elixir, ao executar este script, irá ignorar tudo de #
até o fim da linha, tratando-a como dados ocultos e sem lógica de execução. Pode adicionar nenhum valor para a operação ou alterar o desempenho do script, no entanto, quando não é tão óbvio o que está acontecendo, um programador deve conseguir entender ao ler o comentário. Esteja atento para não abusar do comentário de uma linha! Bagunçar uma base de código pode se tornar um pesadelo indesejável para alguns. É melhor usar com moderação.
Documentação de Módulos
A anotação @moduledoc
permite a documentação em linha em um nível de módulo. É tipicamente situada logo abaixo da declaração defmodule
no topo de um arquivo. O exemplo abaixo mostra um comentário de uma linha dentro do decorador @moduledoc
.
defmodule Greeter do
@moduledoc """
Provides a function `hello/1` to greet a human
"""
def hello(name) do
"Hello, " <> name
end
end
Nós (ou outros) podemos acessar esta documentação de módulo usando a função h
helper dentro de IEx. Nós podemos ver por nós mesmos se colocarmos nosso módulo Greeter
em um novo arquivo chamado greeter.ex
e compilarmos:
iex> c("greeter.ex", ".")
[Greeter]
iex> h Greeter
Greeter
Provides a function hello/1 to greet a human
Nota: não precisamos compilar manualmente nossos arquivos como fizemos acima se estamos trabalhando dentro do contexto de um projeto mix. Você pode usar o iex -S mix
para carregar o IEx console de um projeto atual se você estiver trabalhando em um projeto mix.
Documentação de Funções
Assim como Elixir nos dá a capacidade para anotação em nível de módulo, ele também permite anotações semelhantes para documentar funções. A anotação @doc
permite a documentação de funções. A anotação @doc
fica logo acima da função que está anotando.
defmodule Greeter do
@moduledoc """
...
"""
@doc """
Prints a hello message
## Parameters
- name: String that represents the name of the person.
## Examples
iex> Greeter.hello("Sean")
"Hello, Sean"
iex> Greeter.hello("pete")
"Hello, pete"
"""
@spec hello(String.t()) :: String.t()
def hello(name) do
"Hello, " <> name
end
end
Se utilizarmos IEx novamente e executar o comando auxiliar (h
) sobre a função prefixada com o nome do módulo, devemos ver o seguinte.
iex> c("greeter.ex", ".")
[Greeter]
iex> h Greeter.hello
def hello(name)
@spec hello(String.t()) :: String.t()
Prints a hello message.
Parameters
• name: String that represents the name of the person.
Examples
iex> Greeter.hello("Sean")
"Hello, Sean"
iex> Greeter.hello("pete")
"Hello, pete"
iex>
Observe como você pode usar marcação na documentação e o terminal irá renderizar isto. Além de realmente ser uma ótima nova adição para o vasto ecossistema de Elixir, ficará muito mais interessante quando olharmos o ExDoc para gerar documentação HTML em tempo real.
Nota: a anotação @spec
é usada para analisar estaticamente o código. Para aprender mais sobre isso, veja a lição Especificações e tipos.
ExDoc
ExDoc é um projeto oficial do Elixir que produz HTML (Hyper Text Markup Language) e documentação online para projetos Elixir que pode ser encontrado no Github. Primeiro vamos criar um projeto Mix para a nossa aplicação:
$ mix new greet_everyone
* creating README.md
* creating .formatter.exs
* creating .gitignore
* creating mix.exs
* creating lib
* creating lib/greet_everyone.ex
* creating test
* creating test/test_helper.exs
* creating test/greet_everyone_test.exs
Your Mix project was created successfully.
You can use "mix" to compile it, test it, and more:
cd greet_everyone
mix test
Run "mix help" for more commands.
$ cd greet_everyone
Agora, copie e cole o código a partir da lição de anotação @doc
dentro do arquivo chamado lib/greeter.ex
e garanta que tudo ainda está funcionando na linha de comando. Agora que estamos trabalhando dentro do projeto Mix nós temos que iniciar o IEx um pouco diferente usando o comando iex -S mix
:
iex> h Greeter.hello
def hello(name)
@spec hello(String.t()) :: String.t()
Prints a hello message.
Parameters
• name: String that represents the name of the person.
Examples
iex> Greeter.hello("Sean")
"Hello, Sean"
iex> Greeter.hello("pete")
"Hello, pete"
Instalando
Assumindo que tudo está bem, a saída acima sugere que estamos prontos para configurar o ExDoc. Dentro do nosso arquivo mix.exs
adicione a dependência necessária para começar: :ex_doc
.
def deps do
[{:ex_doc, "~> 0.21", only: :dev, runtime: false}]
end
Nós especificamos o par de chave-valor only :dev
, já que não desejamos fazer o download e compilar essas dependências em um ambiente de produção. O ex_doc
irá também adicionar outra biblioteca para nós, chamada Earmark. Earmark é um parser para Markdown da linguagem de programação Elixir no qual ExDoc utiliza para converter nossa documentação dentro de @moduledoc
e @doc
em uma bela estrutura HTML.
É interessante notar neste momento que você pode mudar a ferramenta de marcação para a Cmark caso desejar, mas você precisará fazer um pouco mais de configuração e você pode ler sobre aqui. Para este tutorial, vamos continuar utilizando Earmark.
Gerando Documentação
Prosseguindo, a partir da linha de comando execute os dois comandos a seguir:
$ mix deps.get # gets ExDoc + Earmark.
$ mix docs # makes the documentation.
Docs successfully generated.
View them at "doc/index.html".
Com esperança, se tudo correu como planejado, você deve ver uma mensagem semelhante como a mensagem de saída no exemplo acima. Vamos agora olhar para dentro do nosso projeto Mix e devemos ver que há um outro diretório chamado doc/. Dentro estará nossa documentação gerada. Se visitarmos a página index em nosso navegador devemos ver o seguinte:
Podemos ver que Earmark converteu nosso markdown e o ExDoc está exibindo-o em um formato útil.
Agora nós podemos implantar isso para GitHub, o nosso próprio site, ou mais comumente no HexDocs.
Boas Práticas
Documentação deve ser adicionada seguindo as boas práticas orientadas pela linguagem. Considerando que Elixir é uma linguagem bastante jovem, muitas normas ainda serão descobertas ao longo do crescimento do ecossistema. A comunidade, entretanto, tem feito esforços para estabelecer as melhores práticas. Para ler mais sobre, veja O Guia de Estilo Elixir.
- Sempre documente um módulo.
defmodule Greeter do
@moduledoc """
This is good documentation.
"""
end
-
Caso você não pretenda documentar um módulo, não deixe-o em branco. Considere anotar o módulo com
false
como a seguir:
defmodule Greeter do
@moduledoc false
end
- Quando se referir a funções dentro da documentação de um módulo, use backticks desta forma:
defmodule Greeter do
@moduledoc """
...
This module also has a `hello/1` function.
"""
def hello(name) do
IO.puts("Hello, " <> name)
end
end
-
Separe todo e qualquer código de uma única linha abaixo de
@moduledoc
como a seguir:
defmodule Greeter do
@moduledoc """
...
This module also has a `hello/1` function.
"""
alias Goodbye.bye_bye
# and so on...
def hello(name) do
IO.puts("Hello, " <> name)
end
end
- Use markdown dentro de funções para torná-lo mais fácil de ler, até em caso de leitura através de IEx ou ExDoc.
defmodule Greeter do
@moduledoc """
...
"""
@doc """
Prints a hello message
## Parameters
- name: String that represents the name of the person.
## Examples
iex> Greeter.hello("Sean")
"Hello, Sean"
iex> Greeter.hello("pete")
"Hello, pete"
"""
@spec hello(String.t()) :: String.t()
def hello(name) do
"Hello, " <> name
end
end
-
Tente incluir alguns exemplos de código em sua documentação, isto também permite gerar testes automáticos a partir dos exemplos de código encontrados em um módulo, função ou macro com Exunit.DocTest. A fim de fazer isso, é preciso invocar o macro
doctest/1
de seu caso de teste e escrever os seus exemplos de acordo com algumas orientações, que são detalhados na documentação oficial.
Caught a mistake or want to contribute to the lesson? Edit this lesson on GitHub!