Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adiciona validação do RENAVAM - Issue 430 #440

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

- Utilitário `convert_code_to_uf` [#397](https://github.com/brazilian-utils/brutils-python/pull/410)

- Utilitário `is_valid_renavam` [#430](https://github.com/brazilian-utils/brutils-python/pull/440)

## [2.2.0] - 2024-09-12

### Added
Expand Down
35 changes: 34 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,8 @@ False
- [generate_voter_id](#generate_voter_id)
- [IBGE](#ibge)
- [convert_code_to_uf](#convert_code_to_uf)
- [RENAVAM](#renavam)
- [is_valid_renavam](#is_valid_renavam)

## CPF

Expand Down Expand Up @@ -1109,6 +1111,37 @@ Exemplo:
>>>
```

## RENAVAM

### is_valid_renavam
Valida o número de registro dos veículos brasileiros (RENAVAM).

Essa função recebe a string do RENAVAM e veirifíca se está válido.
Um renavam válido é constituído por 11 digitos, onde o último é o digito verificador que é calculado por meio dos 10 digitos anteriores.

Args:
* renavam (str): A string do RENAVAM para ser validado.

Retorna:
* bool: Verdadeiro caso o RENAVAM seja válido, Falso caso contrário.

Exemplo:
```python
>>> is_valid_renavam('12345678900')
True
>>> is_valid_renavam('12345678901')
False
>>> is_valid_renavam('1234567890a')
False
>>> is_valid_renavam('12345678 901')
False
>>> is_valid_renavam('12345678') # Less than 11 digits
False
>>> is_valid_renavam('') # Empty string
False
>>> is_valid_renavam(None) # None
False
```

# Novos Utilitários e Reportar Bugs

Expand Down Expand Up @@ -1139,4 +1172,4 @@ Vamos construir juntos! 🚀🚀
[github-discussions-doc]: https://docs.github.com/pt/discussions
[github-discussions]: https://github.com/brazilian-utils/brutils-python/discussions
[github-issues-doc]: https://docs.github.com/pt/issues/tracking-your-work-with-issues/creating-an-issue
[github-issues]: https://github.com/brazilian-utils/brutils-python/issues
[github-issues]: https://github.com/brazilian-utils/brutils-python/issues
38 changes: 36 additions & 2 deletions README_EN.md
Original file line number Diff line number Diff line change
Expand Up @@ -89,7 +89,8 @@ False
- [generate_voter_id](#generate_voter_id)
- [IBGE](#ibge)
- [convert_code_to_uf](#convert_code_to_uf)

- [RENAVAM](#renavam)
- [is_valid_renavam](#is_valid_renavam)
## CPF

### is_valid_cpf
Expand Down Expand Up @@ -1112,6 +1113,39 @@ Exemplo:
>>>
```

## RENAVAM

### is_valid_renavam
Validates the Brazilian vehicle registration number (RENAVAM).

This function takes a RENAVAM string and checks if it is valid.
A valid RENAVAM consists of exactly 11 digits, with the last digit as
a verification digit calculated from the previous 10 digits.

Args:
* renavam (str): The RENAVAM string to be validated.

Returns:
* bool: True if the RENAVAM is valid, False otherwise.

Example:
```python
>>> is_valid_renavam('12345678900')
True
>>> is_valid_renavam('12345678901')
False
>>> is_valid_renavam('1234567890a')
False
>>> is_valid_renavam('12345678 901')
False
>>> is_valid_renavam('12345678') # Less than 11 digits
False
>>> is_valid_renavam('') # Empty string
False
>>> is_valid_renavam(None) # None
False
```

# Feature Request and Bug Report

If you want to suggest new features or report bugs, simply create
Expand Down Expand Up @@ -1141,4 +1175,4 @@ Let's build it together 🚀🚀
[github-discussions-doc]: https://docs.github.com/en/discussions
[github-discussions]: https://github.com/brazilian-utils/brutils-python/discussions
[github-issues-doc]: https://docs.github.com/en/issues/tracking-your-work-with-issues/creating-an-issue
[github-issues]: https://github.com/brazilian-utils/brutils-python/issues
[github-issues]: https://github.com/brazilian-utils/brutils-python/issues
7 changes: 7 additions & 0 deletions brutils/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,11 @@
remove_symbols as remove_symbols_pis,
)

# RENAVAM Imports
from brutils.renavam import (
is_valid_renavam,
)

# Voter ID Imports
from brutils.voter_id import (
format_voter_id,
Expand Down Expand Up @@ -172,4 +177,6 @@
"is_valid_voter_id",
# IBGE
"convert_code_to_uf",
# RENAVAM
"is_valid_renavam",
]
47 changes: 47 additions & 0 deletions brutils/renavam.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
def is_valid_renavam(renavam): # type: (str) -> bool
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ao invés de tipar os parâmetros e o retorno como comentário, utilize mesmo como se fosse uma tipagem forte em linguagens compiladas.

Suggested change
def is_valid_renavam(renavam): # type: (str) -> bool
def is_valid_renavam(renavam: str) -> bool:

"""
Validates the Brazilian vehicle registration number (RENAVAM).

This function takes a RENAVAM string and checks if it is valid.
A valid RENAVAM consists of exactly 11 digits, with the last digit as
a verification digit calculated from the previous 10 digits.

Args:
renavam (str): The RENAVAM string to be validated.

Returns:
bool: True if the RENAVAM is valid, False otherwise.

Example:
>>> is_valid_renavam('12345678900')
True
>>> is_valid_renavam('12345678901')
False
>>> is_valid_renavam('1234567890a')
False
>>> is_valid_renavam('12345678 901')
False
>>> is_valid_renavam('12345678') # Less than 11 digits
False
>>> is_valid_renavam('') # Empty string
False
>>> is_valid_renavam(None) # None
False
"""
if renavam and len(renavam) == 11 and renavam.isnumeric():
check_digit = int(renavam[-1:])
renavam_digitis = [int(d) for d in renavam]
renavam_digitis = renavam_digitis[:-1]
multipliers = [2, 3, 4, 5, 6, 7, 8, 9, 2, 3]
sum_digits = sum(
digit * multipliers[i]
for i, digit in enumerate(renavam_digitis[::-1])
)
remainder_division = sum_digits % 11
if remainder_division <= 1:
check_digit_calculated = 0
else:
check_digit_calculated = 11 - remainder_division
if check_digit == check_digit_calculated:
return True
return False
Comment on lines +34 to +47
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Na codebase existe essa funcção que está implementada no PIS:

def _checksum(base_pis: str) -> int:
    """
    Calculate the checksum digit of the given `base_pis` string.

    Args:
        base_pis (str): The first 10 digits of a PIS number as a string.

    Returns:
        int: The checksum digit.
    """
    pis_digits = list(map(int, base_pis))
    pis_sum = sum(digit * weight for digit, weight in zip(pis_digits, WEIGHTS))
    check_digit = 11 - (pis_sum % 11)

    return 0 if check_digit in [10, 11] else check_digit

Me parece que todo documento utilizado no Brasil (PIS, PASEP, RENAVAM, CPF) utilizam esse código de maneira parecida. Então, o que eu sugiro é você utilizar essa função que já existes ajustando para o que você precisa, assim, a gente evita muita duplicação de código.

38 changes: 38 additions & 0 deletions tests/test_renavam.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
from unittest import TestCase

from brutils.renavam import is_valid_renavam


class TestRENAVAM(TestCase):
def test_is_valid_renavam(self):
# Testes para RENAVAM válidos
self.assertFalse(is_valid_renavam("12345678901"))
self.assertFalse(is_valid_renavam("09945438645"))
self.assertFalse(is_valid_renavam("94521237655"))
self.assertFalse(is_valid_renavam("45403264305"))
self.assertFalse(is_valid_renavam("45403471665"))
self.assertFalse(is_valid_renavam("34743721835"))
self.assertFalse(is_valid_renavam("69277208515"))

self.assertTrue(is_valid_renavam("12345678900"))
self.assertTrue(is_valid_renavam("92876838150"))
self.assertTrue(is_valid_renavam("65720648534"))
self.assertTrue(is_valid_renavam("63601073019"))
self.assertTrue(is_valid_renavam("09945438641"))
self.assertTrue(is_valid_renavam("94521237651"))
self.assertTrue(is_valid_renavam("45403264308"))
self.assertTrue(is_valid_renavam("45403471664"))
self.assertTrue(is_valid_renavam("34743721831"))
self.assertTrue(is_valid_renavam("69277208510"))

# Testes para entradas inválidas
self.assertFalse(is_valid_renavam("1234567890a")) # Contém letra
self.assertFalse(is_valid_renavam("12345678 901")) # Contém espaço
self.assertFalse(is_valid_renavam("12345678")) # Menos de 11 dígitos
self.assertFalse(is_valid_renavam("")) # String vazia
self.assertFalse(is_valid_renavam("123456789012")) # Mais de 11 dígitos
self.assertFalse(is_valid_renavam("abcdefghijk")) # Apenas letras
self.assertFalse(
is_valid_renavam("12345678901!")
) # Contém caractere especial
self.assertFalse(is_valid_renavam(None)) # Contém caractere especial