# LazyDoc
<p align='center'>
<a href="https://pxp9.github.io/lazy-doc/"><img alt="Static Badge" src="https://img.shields.io/badge/GitHub%20Pages-222222?style=for-the-badge&logo=github&logoColor=white&label=Read%20the%20article!&link=https%3A%2F%2Fpxp9.github.io%2Flazy-doc"></a>
<a href="https://elixirforum.com/t/lazydoc-detect-undocumented-functions-and-pass-the-function-to-an-ai-provider-to-document-it/69818"><img alt="Static Badge" src="https://img.shields.io/badge/elixir-%25234B275F.svg?style=for-the-badge&logo=elixir&logoColor=white&label=Read%20the%20post!&color=purple&link=https%3A%2F%2Felixirforum.com%2Ft%2Flazydoc-detect-undocumented-functions-and-pass-the-function-to-an-ai-provider-to-document-it%2F69818"></a>
</p>
Lazy Doc is a project for those who are lazy af to document their code.
It is designed to detect undocumented functions, pass the function to an AI
provider which is a tuple of three elements `{GithubAi, :codestral, options}`.
## Installation
``` elixir
def deps do
[
{:lazy_doc, "~> 0.6.1"}
]
end
```
or install it as a single binary
``` bash
mix escript.install hex lazy_doc 0.6.1
```
## Configuration
Configuration is the same if you are using the binary or the dep.
`config/lazy_doc.exs` It is expected to be in this file with Elixir config
syntax.
``` elixir
import Config
alias LazyDoc.Providers.GithubAi
## configure formatter.
config :lazy_doc,
## This is the default value, no need to configure.
## This option creates the docs in files under priv/lazy_doc directory.
external_docs: false,
## This is the default value, no need to configure.
patterns: [
~r"^lib/[a-zA-Z_]+(?:/[a-zA-Z_]+)*/[a-zA-Z_]+\.ex$",
],
## This is the default value, no need to configure.
max_retries: 1,
## This is the default value, no need to configure.
receive_timeout: 15_000,
## This is the default value, no need to configure.
file_formatter: ".formatter.exs",
## This is the default value, no need to configure.
provider: {GithubAi, :gpt_4o_mini, [max_tokens: 2048, top_p: 1, temperature: 1]},
## This is the default value, no need to configure.
custom_function_prompt:
~s(You should describe the parameters based on the spec given and give a small description of the following function.\n\nPlease do it in the following format given as an example, important do not return the header of the function, do not return a explanation of the function, your output must be only the docs in the following format.\n\n@doc """\n\n## Parameters\n\n- transaction_id - foreign key of the Transactions table.\n## Description\n Performs a search in the database\n\n## Returns\n the Transaction corresponding to transaction_id\n\n"""\n\nFunction to document:\n),
## This is the default value, no need to configure.
custom_module_prompt:
~s(You should describe what this module does based on the code given.\n\n Please do it in the following format given as an example, important do not return the code of the module, your output must be only the docs in the following format.\n\n@moduledoc """\n\nThe module GithubAi provides a way of communicating with Github AI API \(describes the main functionality of the module\).\n\n## Description\n\nIt implements the behavior Provider a standard way to use a provider in LazyDoc.\(gives a detailed description of what the module does\)\n"""\n\nModule to document:\n)
```
> Note If installed as a dep you need to do this.
`config/config.exs`
``` elixir
import_conf("lazy_doc.exs")
```
### API token conf
`.env`
``` bash
API_TOKEN="YOUR AWESOME TOKEN"
```
if installed as dep
`config/runtime.exs`
``` elixir
## Required to be configured
config :lazy_doc, :token, System.get_env("API_TOKEN")
```
if installed as binary
``` bash
export API_TOKEN="YOUR AWESOME TOKEN"
```
or
``` bash
export $(cat .env | xargs)
```
By default `lazy_doc` will try to get the token from the env var `API_TOKEN`
### Available Providers implemented by LazyDoc
You can check Providers implemented in LazyDoc
[here](https://hexdocs.pm/lazy_doc/github_ai.html).
## How to run it ?
### As a Elixir dependency
From the root of the elixir project once installed and configured.
``` bash
mix lazy_doc
```
If you want, you can add a simple check to see what needs to be documented in
your project. This is good for CI.
``` bash
mix lazy_doc.check
```
### As a binary
``` bash
lazy_doc
```
``` bash
lazy_doc --check
```
## Build the binary from source and install it.
``` bash
MIX_ENV=dev mix escript.build
```
``` bash
MIX_ENV=dev mix escript.install
```
## Known limitations that wont be fixed.
### Module names in the same file must be different.
If the user creates an inner module with the same name as the parent module
`lazy_doc`, it wont work properly because they have the same `:__aliases__` AST
node.
> \[\!WARNING\] This limitation it is only in module names. So if the user have
> same names of functions in different modules or in the same module, it will
> work.
``` elixir
defmodule Hello do
defmodule Hello do
end
end
```
produces the following AST
``` elixir
{:defmodule, [line: 1],
[
## This is why it does not work [:Hello]
{:__aliases__, [line: 1], [:Hello]},
[
do: {:defmodule, [line: 3],
[
{:__aliases__, [line: 3],
## This is why it does not work [:Hello]
[:Hello]},
[do: {:__block__, [], []}]
]}
]
]}
```
## Roadmap
- [X] Make AI docs for functions `@doc`.
- [X] Simple check if the response is in `@doc` format.
- [X] Make AI providers more extensible (define a behavior, what an AI provider
should do ?).
- [X] Custom path wildcard (limits the action of `lazy_doc`)
- [X] Make some unit tests.
- [X] Improve the default prompt to generate markdown syntax.
- [X] Fix inner module detection (creates scopes for inner modules and builds
the full name of the inner module).
- [X] Make a task or an arg in the current task to check if the functions are
documented. (allows CI usage)
- [X] File is written to file according to Elixir formatter.
- [X] Make AI docs for modules as well, `@moduledoc`.
- [X] Custom prompts for function and module.
- [ ] Simple check if the response is in `@moduledoc` format.
- [X] Customizable number of retries.
- [X] Custom paramters to pass the provider (max\_tokens, top\_p, temperature).
- [X] Check if custom paramters are valid for that provider.
- [ ] Inspect the `defimpl` and `defprotocol` nodes.
- [ ] Make `@moduledoc` documentation take data from the Modules is related the
given module.
- [ ] Spec mode: Detect if a function does not have any `@specs` and try to
generate their specs.
- [X] Make an option `external_docs` to do `@doc
File.read!("file_gen_by_lazy_doc.md")`, this will prevent code redability
issues, it will create a folder where all the docs is generated.
- [X] Make a `mix lazy_doc.clean` task that will allow to clear all the docs for
updating to new docs.
- [X] Make lazy\_doc run in a single binary