Gerenciamento de dependência em Go (Go Modules)
Go Modules
Cada linguagem de programação possui uma maneira de instalar dependências/bibliotecas e gerenciar as versões de cada uma dessas dependências no projeto. A maneira com que o Go gerencia suas dependências desde a versão 1.11 é utilizando Módulos (Go Modules). Em versões passadas do Go as dependências eram gerenciadas com ferramentas auxiliares como o dep e é possível migrar para o Go Modules, mais informações sobre como fazer isso pode ser vistas na wiki do Go.
Para iniciar um novo projeto em Go utilizando o Go Modules vá até a raíz do projeto e execute o comando go mod init meu-modulo
. Nesse comando o meu-modulo
pode ter o caminho completo para um repositório de um sistema de controle de versionamento, por exemplo , github.com/brenoassp/meu-modulo
.
O comando go mod init
gera um arquivo com nomego.mod
que contém o nome do módulo criado junto com a versão do Go utilizada, como mostrado abaixo:
A medida que novos pacotes forem sendo adicionados no projeto esse arquivo vai sendo modificado com os pacotes instalados, por exemplo, podemos instalar um novo pacote com o comando a seguir:
go get -u github.com/valyala/fasthttp
Após isso o arquivo go.mod
terá o seguinte conteúdo:
Apesar de ter instalado apenas uma dependência, mais de um pacote foi instalado, isso acontece porque geralmente todo pacote que instalamos depende também de outros pacotes. Como esses pacotes internos das dependências que instalamos não são utilizados diretamente por nós eles são marcados como dependência indireta com o comentário indirect
depois da versão. Nesse caso específico, até mesmo a dependência do fasthttp que instalamos está sendo marcada como dependência indireta pois não estamos usando ela em nenhum lugar no código, apesar de ter sido instalada.
Ao criar um código que utilize o fasthttp como o código abaixo, ele deixará de ser uma dependência indireta no nosso projeto.
O comando go mod tidy
é responsável por atualizar o go.mod
seja atualizando ou removendo dependências. Ao executar esse comando o go.mod
passa a ter o seguinte conteúdo:
Ou seja, o fasthttp deixa de ser uma dependência indireta para ser uma dependência direta já que está sendo utilizado. Se removermos o código que utiliza o fasthttp e executarmos o go mod tidy
novamente ele removerá do go.mod
as dependências pois elas não estão sendo utilizadas em nenhum momento.
Além do go.mod
existe também o arquivo go.sum
que contém todas as dependências junto com a hash que representa o conteúdo do pacote. Isso serve pra garantir que os pacotes baixados futuramente no projeto são iguais aos pacotes baixados no momento em que a dependência foi adicionada no projeto. Ambos os arquivos devem ser versionados e é recomendável usar o comando go mod tidy
regularmente para manter esses arquivos atualizados somente com as dependências que estão realmente sendo utilizadas.
Não é necessário ficar utilizando o go get
para adicionar todos os pacotes que serão utilizados pois ao compilar o projeto (go build
) ou executar os testes (go test
) as dependências serão baixadas automaticamente com base nas importações feitas no início dos arquivos com extensão .go
e o go.mod
é atualizado automaticamente.
Durante o desenvolvimento é comum que algumas dependências deixem de ser necessárias, mas essas dependências não são removidas automaticamente dos arquivo go.mod
e go.sum
durante a compilação ou execução dos testes. Nesse caso, é preciso executar o comando go mod tidy
que irá verificar as dependências que não são mais necessárias e removê-las desses arquivos.
Diferentemente de vários gerenciadores de dependências de outras linguagens, o Go Module não instala as dependências do projeto em uma pasta contida na raiz do repositório. O que controla onde será salvo as bibliotecas são as variáveis de ambiente GOPATH e GOBIN. Caso elas não estejam definidas a pasta padrão utilizada é: /home/[user]/go
.
GOBIN: indica onde serão instalados os binários.
GOPATH: indica onde serão instalados os pacotes. Se essa variável existir os binários são instalados na pasta bin
do primeiro diretório da lista de diretórios existentes nessa variável.
Os módulos são versionados utilizando a especificação de versionamento semântico SemVer. Em resumo, essa especificação diz que a nomenclatura do pacote deve ser do tipo X.Y.Z, onde:
X: Número que representa a versão principal do pacote (Major Version). É incrementado somente quando há quebra de compatibilidade na API.
Y: Número que representa a versão secundária (Minor Version). É incrementada quando há adição de novas funcionalidades, mas sem quebrar a compatibilidade com a versão anterior.
Z: Número que representa a versão de correção (Patch Version). É incrementada quando é realizado alguma correção de bug.
Acredito que o mais importante a se saber é que, por padrão, o Go não atualiza sozinho a major version dos pacotes ao utilizar comandos como go get -u modulo
, ele apenas atualiza para a versão mais recente da minor version e patch version. Quando existe uma atualização com quebra de compatibilidade em algum módulo é possível ver de forma explícita pois é necessário indicar essa versão ao importar o pacote dentro do código da seguinte forma:
import "github.com/brenoassp/meu-modulo/v2"
De qualquer forma, sempre que for atualizada alguma biblioteca no projeto é interessante rodar os testes do projeto para garantir que nada foi quebrado, independentemente da atualização que foi feita.
TL;DR;
Comando | Exemplo | Descrição |
---|---|---|
go mod init [path] | go mod init github.com/brenoassp/meu-modulo | Inicializa um novo módulo no diretório corrente criando um arquivo go.mod nesse local |
go mod tidy | - | Atualiza o go.mod seja adicionando dependências que estão faltando ou removendo as que não estão sendo utilizadas |
go get modulo | go get https://github.com/valyala/fasthttp | Baixa ou atualiza o pacote fasthttp na versão mais atualizada |
go get -u modulo | go get -u https://github.com/valyala/fasthttp | Atualiza tanto o pacote fasthttp e as suas dependências para a versão mais atualizada |
go mod download | - | Baixa para o cache local todos os módulos existentes no arquivo go.mod |