Obrigado por dedicar o seu tempo para contribuir! 🙇♀️🙇♂️ Toda ajuda é bem-vinda!
Como fazer a sua primeira contribuição:
- 1. Crie uma Conta no GitHub
- 2. Encontre uma Issue para Trabalhar
- 3. Instale o Git
- 4. Faça um Fork do Projeto
- 5. Clone o Seu Fork
- 6. Crie um Novo Branch
- 7. Execute o brutils Localmente
- 8. Faça as Suas Alterações
- 9. Teste as Suas Alterações
- 10. Atualizar READMEs
- 11. Faça o Commit e Envie as Suas Alterações
- 12. Adicione Entradas no CHANGELOG.md
- 13. Crie um PR no GitHub
- 14. Atualizar a Sua Branch se Necessário
Certifique-se de ter uma conta no GitHub e de estar com a sessão iniciada.
Caso não tenha uma conta, siga os passos de como criar de uma conta pessoal no GitHub.
Visite a página de issues do brutils e encontre uma issue com a qual você gostaria de trabalhar e que ainda não tenha sido atribuída a ninguém.
Deixe um comentário na issue com conteúdo "bora!" Em seguida, um bot vai atribuir a issue a você. Uma vez atribuída, você pode prosseguir para a próxima etapa.
Sinta-se à vontade para fazer qualquer pergunta na página da issue antes ou durante o processo de desenvolvimento.
Ao começar a contribuir para o projeto, é recomendável que você pegue uma issue por vez. Isso ajuda a garantir que outras pessoas também tenham a oportunidade de colaborar e evita que recursos fiquem inativos por muito tempo.
Certifique-se de ter o git instalado, seguindo os passos do tutorial de instalação do git.
Faça um fork do repositório brutils.
Clone o seu fork localmente.
Entre na pasta do brutils:
$ cd brutils-python
>
E crie uma nova branch com o nome da issue em que você irá trabalhar através do comando:
$ git checkout -b <issue_number>
>
Exemplo:
$ git checkout -b 386
Switched to a new branch '386'
>
Crie um virtualenv para o brutils e o ative através do comando:
$ make shell
Spawning shell within ...-py3.x
emulate bash -c '. .../bin/activate'
Para testar se o ambiente virtual está ativo corretamente, execute o comando e verifique se a resposta é algo parecido com a seguinte:
$ poetry env info
Virtualenv
Python: 3.x.y
Implementation: CPython
...
Observação: Você precisa executar make shell
toda vez que abrir uma nova janela ou aba do terminal.
Instale as dependências:
$ make install
git config --local core.hooksPath .githooks/
chmod -R +x .githooks
Installing dependencies from lock file
...
Se preferir usar pip, você pode instalar o projeto em modo de desenvolvimento da seguinte forma:
- Python 3.8+
- [pip][pip]
Crie um virtualenv para o brutils e o ative através do comando:
python -m venv venv
source venv/bin/activate # No Windows use: venv\Scripts\activate
Utilize o comando pip para instalar as dependencias de dev e testes através do arquivo requirements-dev.txt
pip install -r requirements-dev.txt
Agora você pode usá-lo da mesma forma descrita no arquivo README.md.
Execute os testes através do seguinte comando:
$ make test
make test
test__is_valid_mercosul (license_plate.test_is_valid.TestIsValidMercosul.test__is_valid_mercosul) ... ok
test__is_valid_old_format (license_plate.test_is_valid.TestIsValidOldFormat.test__is_valid_old_format) ... ok
....
----------------------------------------------------------------------
Ran XX tests in 0.000s
OK
Certifique-se de que o retorno é OK
, o quê indica todos os testes estão passando e que não tem nenhum falhando.
Agora é a etapa em que você pode implementar as suas alterações no código.
Normalmente existem instruções/ideias de como você pode implementar a solução diretamente na descrição da issue, na seção "Descreva alternativas que você considerou". Leia atentamente tudo que está escrito na issue para garantir que suas modificações resolvem tudo que está sendo solicitado.
É importante notar que documentamos o nosso código usando docstrings. Módulos, classes, funções e métodos devem ser documentados. Suas alterações também devem ser bem documentadas e refletir docstrings atualizadas, caso algum dos parâmetros tenha sido alterado para um classe/atributo ou mesmo funções.
Todas as docstring devem estar em Inglês. Fique à vontade para utilizar ferramentas como Google Tradutor ou ChatGPT caso precise. Iremos sugerir mudanças na tradução se necessário, então não se preocupe com possíveis erros de inglês.
Seguimos o padrão abaixo para manter consistência nas docstrings:
class Example:
"""
Explain the purpose of the class
Attributes:
x[dict]: Short explanation here
y[type, optional]: Short explanation here
"""
def __init__(self, x, y=None):
self.x = x
self.y = y
def foobar(self, w):
"""
Purpose of the function
Args:
name[type]: Short explanation here
Returns:
type: Short explanation here
Example:
>>> command 1
output 1
>>> command 2
output 2
"""
...
return value
Exemplo:
def format_cep(cep): # type: (str) -> str | None
"""
Formats a Brazilian CEP (Postal Code) into a standard format.
This function takes a CEP (Postal Code) as input and, if it is a valid
8-digit CEP, formats it into the standard "12345-678" format.
Args:
cep (str): The input CEP (Postal Code) to be formatted.
Returns:
str: The formatted CEP in the "12345-678" format if it's valid,
None if it's not valid.
Example:
>>> format_cep("12345678")
"12345-678"
>>> format_cep("12345")
None
"""
return f"{cep[:5]}-{cep[5:8]}" if is_valid(cep) else None
Algo a se ter em mente ao documentar o código com docstrings é que você pode ignorar docstrings em decoradores de propriedade e métodos mágicos.
Certifique-se de ter criado os testes necessários para cada nova alteração que você fez.
Execute todos os testes com o comando make test
e certifique-se de que todos passam.
Os PRs não serão mesclados se houver algum teste faltando ou falhando.
Abra um ambiente interativo para testar manualmente as suas mudanças:
$ python
Python 3.x.y ...
Type "help", "copyright", "credits" or "license" for more information.
>>> # Teste as suas mudanças aqui!
Exemplo:
$ python
Python 3.12.5 (main, Aug 6 2024, 19:08:49) [Clang 15.0.0 (clang-1500.3.9.4)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> import brutils
>>> from brutils import cpf
>>> cpf.generate()
'13403202232'
>>> from brutils import generate_cpf
>>> generate_cpf()
'64590379228'
Atualize os arquivos brutils-python/README.md
e brutils-python/README_EN.md
com suas alterações.
Esses arquivos são essenciais para a documentação da biblioteca, ajudando os usuários a entender como utilizar os recursos oferecidos. Portanto, é importante mantê-los sempre atualizados.
- O arquivo brutils-python/README_EN.md contém a documentação em Inglês, e pode usar o conteúdo já descrito na docstring diretamente.
- O arquivo brutils-python/README.md contém a documentação em Português. Para este, basta traduzir a docstring da função.
Se precisar de assistência na tradução para o Português, ferramentas como Google Tradutor ou ChatGPT podem ajudar. Não se preocupe com possíveis erros de tradução, pois sugeriremos ajustes quando necessário.
Exemplo em Inglês (README_EN.md):
### format_cep
This function takes a CEP (Postal Code) as input and, if it is a valid
8-digit CEP, formats it into the standard "12345-678" format.
Args:
- cep (str): The input CEP (Postal Code) to be formatted.
Returns:
- str: The formatted CEP in the "12345-678" format if it's valid,
None if it's not valid.
Example:
```python
>>> from brutils import format_cep
>>> format_cep('01310200')
'01310-200'
>>> format_cep("12345678")
"12345-678"
>>> format_cep("12345")
None
```
Exemplo em Português (README.md):
### format_cep
Formata um CEP (Código de Endereçamento Postal) brasileiro em um formato padrão.
Esta função recebe um CEP como entrada e, se for um CEP válido com 8 dígitos,
o formata no padrão "12345-678".
Argumentos:
- cep (str): O CEP (Código de Endereçamento Postal) de entrada a ser
formatado.
Retorna:
- str: O CEP formatado no formato "12345-678" se for válido, None se não for
válido.
Example:
```python
>>> from brutils import format_cep
>>> format_cep('01310200')
'01310-200'
>>> format_cep("12345678")
"12345-678"
>>> format_cep("12345")
None
```
Fomarte o seu código executando o comando:
$ make format
...
Exemplo:
$ make format
31 files left unchanged
All checks passed!
Faça o commit das alterações:
$ git commit -a -m "<commit_message>"
...
Exemplo:
$ git commit -m 'Adicionando mais info aos arquivos de contribuição'
[386 173b7e6] Adicionando mais info aos arquivos de contribuição
2 files changed, 144 insertions(+), 34 deletions(-)
Push o seu commit para o GitHub:
$ git push --set-upstream origin <issue_number>
...
Exemplo:
$ git push --set-upstream origin 386
Running pre-push hook checks
All checks passed!
Enumerating objects: 7, done.
Counting objects: 100% (7/7), done.
Delta compression using up to 10 threads
Compressing objects: 100% (4/4), done.
Writing objects: 100% (4/4), 2.36 KiB | 2.36 MiB/s, done.
Total 4 (delta 3), reused 0 (delta 0), pack-reused 0
remote: Resolving deltas: 100% (3/3), completed with 3 local objects.
remote:
remote: Create a pull request for '386' on GitHub by visiting:
remote: https://github.com/brazilian-utils/brutils-python/pull/new/386
remote:
To github.com:brazilian-utils/brutils-python.git
* [new branch] 386 -> 386
Faça as alterações e commits necessários e envie-os quando estiverem prontos.
Um changelog é um arquivo que contém uma lista organizada cronologicamente de mudanças notáveis para cada versão de um projeto.
Para facilitar para usuários e contribuintes verem exatamente quais mudanças notáveis foram feitas entre cada release (ou versão) do projeto.
Pessoas. Sejam consumidores ou desenvolvedores, os usuários finais de software são seres humanos que se importam com o que está no software. Quando o software muda, as pessoas querem saber por que e como.
O changelog do brutils está disponível em CHANGELOG.md
- Changelogs são para humanos, não máquinas.
- Deve haver uma entrada para cada versão.
- Os mesmos tipos de mudanças devem ser agrupados.
- Versões e seções devem ser linkáveis.
- A versão mais recente vem primeiro.
- A data de lançamento de cada versão é exibida.
-
Correções de segurança: Devem ser documentadas com o tipo definido como "segurança" para alertar os usuários sobre questões de segurança resolvidas. Exemplo: “Corrigido um vulnerabilidade crítica que permitia a execução remota de código.”
-
Mudanças voltadas ao usuário: Alterações que afetam diretamente a forma como os usuários interagem com o software, incluindo novas funcionalidades, alterações em funcionalidades existentes ou melhorias na interface. Exemplo: “Adicionada uma nova opção de filtro na página de resultados para facilitar a busca.”
-
Melhorias significativas de desempenho: Devem ser registradas quando resultam em melhorias notáveis na velocidade ou na eficiência do software que impactam a experiência do usuário. Exemplo: “O tempo de carregamento da página inicial foi reduzido em 40% após a otimização do backend.”
-
Alterações que afetam a compatibilidade: Mudanças que ajustam o software para manter a compatibilidade com outras ferramentas, sistemas ou versões. Exemplo: “Atualizada a biblioteca X para a versão 2.0 para suportar a nova versão do Python.”
-
Mudanças na API pública: Alterações que afetam como os desenvolvedores interagem com a API pública do software, seja adicionando novas rotas ou alterando as existentes. Exemplo: “Adicionada uma nova rota /api/v1/users para gerenciamento de usuários.”
-
Alterações nas dependências: Atualizações ou mudanças nas dependências do projeto que podem afetar o comportamento ou a compatibilidade do software. Exemplo: “Atualizado o pacote de dependência Y para a versão 3.1.4, que inclui correções importantes de segurança.”
Embora o changelog seja uma ferramenta valiosa para documentar mudanças, algumas informações não devem ser incluídas. Aqui estão alguns exemplos do que não deve aparecer no changelog:
-
Mudanças Internas de Código: Alterações que não afetam o comportamento do usuário final, como refatorações de código interno que não alteram a funcionalidade, não precisam ser documentadas no changelog. Exemplo: “Refatoração de funções internas” ou “Correção testes inconsistentes.”
-
Melhorias de Desempenho Não Notáveis: Melhorias de desempenho que não resultam em mudanças perceptíveis ou benefícios claros para o usuário final não precisam de uma entrada específica. Exemplo: “Otimização de algoritmos internos.”
-
Correções de Bugs Menores: Correções para bugs que não afetam o uso geral ou a experiência do usuário final podem ser omitidas. Exemplo: “Correção de um pequeno erro de digitação no código.”
-
Mudanças Somente de Documentação: Alterações que afetam apenas a documentação, sem modificar o comportamento do software, geralmente não precisam ser incluídas no changelog. Exemplo: “Atualização do README.md para refletir novas dependências.”
-
Detalhes Técnicos Excessivos: Informações excessivamente técnicas que não são relevantes para o usuário final ou não oferecem contexto sobre o impacto da mudança devem ser evitadas. Exemplo: “Mudança no gerenciamento de memória na classe X.”
-
Entradas de Mantenedor: Mudanças que são relacionadas apenas ao processo de desenvolvimento ou manutenção interna, como ajustes de configuração de ferramentas de CI/CD, geralmente não são relevantes para o changelog. Exemplo: “Atualização na configuração do GitHub Actions.”
-
Uma correção de bug introduzida e corrigida na mesma release não precisa de uma entrada no changelog.
Evite incluir essas informações no changelog para manter o documento focado e útil para os usuários e contribuintes do projeto.
Uma boa entrada no changelog deve ser descritiva e concisa. Deve explicar a mudança a um leitor que não tem nenhum contexto sobre a mudança. Se for difícil ser ao mesmo tempo conciso e descritivo, opte por ser mais descritivo.
- Ruim: Ir para a ordem do projeto.
- Bom: Mostrar os projetos estrelados do usuário no topo do dropdown “Ir para o projeto”.
O primeiro exemplo não dá contexto sobre onde a mudança foi feita, nem por que, nem como beneficia o usuário.
- Ruim: Copiar (algum texto) para a área de transferência.
- Bom: Atualizar o tooltip de “Copiar para a área de transferência” para indicar o que está sendo copiado.
Novamente, o primeiro exemplo é muito vago e não fornece contexto.
- Ruim: Corrige e melhora problemas de CSS e HTML no gráfico de mini pipeline e dropdown de builds.
- Bom: Corrigir tooltips e estados de hover no gráfico de mini pipeline e dropdown de builds.
O primeiro exemplo está muito focado nos detalhes de implementação. O usuário não se importa que mudamos CSS e HTML, ele se importa com o resultado final dessas mudanças.
- Ruim: Remover valores nulos no Array de objetos Commit retornados por find_commits_by_message_with_elastic
- Bom: Corrigir erros 500 causados por resultados do Elasticsearch referenciando commits já recolhidos pelo garbage collector.
O primeiro exemplo foca em como corrigimos algo, não no que foi corrigido. A versão reescrita descreve claramente o benefício final para o usuário (menos erros 500) e quando isso acontece (ao buscar commits com Elasticsearch).
Use seu melhor julgamento e tente se colocar na posição de alguém lendo o changelog compilado. Essa entrada agrega valor? Oferece contexto sobre onde e por que a mudança foi feita?
O changelog está disponível no arquivo CHANGELOG.md.
Primeiro, você precisa identificar o tipo da sua mudança. Tipos de mudanças:
Added
para novas funcionalidades.Changed
para mudanças em funcionalidades existentes.Deprecated
para funcionalidades que serão removidas em breve.Fixed
para qualquer correção de bugs.Removed
para funcionalidades que foram removidas.Security
em caso de vulnerabilidades.
Você deve sempre adicionar novas entradas no changelog na seção Unreleased
. No momento do release, moveremos as mudanças da seção Unreleased
para uma nova seção de versão.
Portanto, dentro da seção Unreleased
, você deve adicionar sua entrada na seção apropriada por tipo. Se ainda não houver uma seção para o tipo da sua mudança, você deve adicioná-la.
Vamos ver alguns exemplos. Suponhamos que você tenha uma nova mudança Fixed
para adicionar, e o arquivo atual do CHANGELOG.md está assim:
## [Unreleased]
### Added
- Utilitário `get_address_from_cep` [#358](https://github.com/brazilian-utils/brutils-python/pull/358)
### Changed
- Utilitário `fmt_voter_id` renomeado para `format_voter_id` [#221](https://github.com/brazilian-utils/brutils-python/issues/221)
Você precisará adicionar uma nova seção Fixed
e incluir a nova entrada lá:
## [Unreleased]
### Added
- Utilitário `get_address_from_cep` [#358](https://github.com/brazilian-utils/brutils-python/pull/358)
### Changed
- Utilitário `fmt_voter_id` renomeado para `format_voter_id` [#221](https://github.com/brazilian-utils/brutils-python/issues/221)
### Fixed
- Minha mensagem de changelog aqui. [#<número_da_issue>](<link_da_issue>)
Note que a ordem das seções por tipo importa. Temos um lint que verifica isso, então as seções devem ser ordenadas alfabeticamente. Primeiro Added
, depois Changed
, terceiro Deprecated
e assim por diante.
Agora, digamos que você tem mais uma entrada para adicionar e o tipo dela é Added
. Como já temos uma seção para isso, você devve apenas adicionar uma nova linha:
## [Unreleased]
### Added
- Utilitário `get_address_from_cep` [#358](https://github.com/brazilian-utils/brutils-python/pull/358)
- Minha outra mensagem de changelog aqui. [#<número_da_issue>](<link_da_issue>)
### Changed
- Utilitário `fmt_voter_id` renomeado para `format_voter_id` [#221](https://github.com/brazilian-utils/brutils-python/issues/221)
### Fixed
- Minha mensagem de changelog aqui. [#<número_da_issue>](<link_da_issue>)
Este conteúdo é baseado no site do keep a changelog, já que seguimos suas diretrizes.
Certifique-se de que sua branch esteja atualizado com o main
Aqui você encontrará como lançar uma nova versão em produção do brutils:
Para a criação da issue, pode ser utilizado o template de feature, sendo o nome da issue Release v<versão>
. Exemplo
O nome da branch criada para o release é relacionado ao número da Issue, como mostra este exemplo
Incremente o número da versão, seguindo o Versionamento Semântico,
no arquivo pyproject.toml
:
- https://github.com/brazilian-utils/brutils-python/blob/main/pyproject.toml#L3
Adicione um título para a nova versão com o novo número e a data atual, como neste exemplo.
E adicione os links da versão, como neste exemplo
Crie um PR com o nome Release v<versão>
contendo as duas alterações acima. Na descrição da Pull Request, adicione o trecho do changelog alterado.
Assim que o PR for aceito e passar em todas as verificações, faça o merge.
O lançamento da nova versão em produção é feita automaticamente quando uma nova release é criada no GitHub.
- Preencha o campo
tag version
com:v<versão>
(por exemplo,v2.0.0
). - Preencha o campo
release title
com o mesmo valor que a versão da tag (por exemplo,v2.0.0
). - Preencha o campo
release description
com o conteúdo copiado do arquivo CHANGELOG.md da seção de versão correspondente.
Exemplos reais estão disponíveis em: https://github.com/brazilian-utils/brutils-python/releases
Quando o Deploy via GitHub for concluído, a nova versão também será lançada automaticamente no PyPI. Baixe a nova versão do brutils do PyPI e teste se tudo está funcionando conforme o esperado.