first commit

This commit is contained in:
Claudecio Martins
2026-06-16 10:04:10 -03:00
commit a951944997
4463 changed files with 419677 additions and 0 deletions
@@ -0,0 +1,111 @@
<?php
namespace WorkbloomERP\Module\v0\Auth\Controllers;
use KrothiumAPI\Utils\HttpUtil;
use WorkbloomERP\Utils\SanitizeUtil;
use WorkbloomERP\Exceptions\AppException;
use WorkbloomERP\Module\v0\Auth\Factories\ServiceFactory;
class AuthController {
/**
* Endpoint de API para autenticação de usuário (**Login**).
*
* Este método gerencia a interface HTTP (POST) para validar as credenciais do usuário.
* Ele atua como uma camada fina (*Thin Controller*), responsável por capturar o
* payload da requisição, sanitizar os dados de entrada para prevenir injeções
* e delegar a verificação de identidade e gestão de sessão/tokens para o `AuthService`.
*
* ---
* ## Fluxo de Execução
* 1. **Captura e Validação:** Extrai o corpo da requisição. Se o payload estiver vazio, interrompe com erro 400 (Bad Request).
* 2. **Sanitização:** Aplica `SanitizeUtil::string()` nos campos de login e senha para garantir a limpeza dos dados antes do processamento de negócio.
* 3. **Processamento de Negócio:** Invoca `AuthService::login()`, que realiza a consulta de credenciais.
* 4. **Normalização de Resposta:**
* - **Sucesso (200 OK):** Retorna os dados de sessão (ex: token de acesso).
* - **Tratamento de Erros:** Captura `AppException` (credenciais inválidas, conta bloqueada, etc.), formatando a resposta para que o frontend receba um objeto de erro estruturado.
*
* ---
* ## Observações Técnicas
* - A responsabilidade de verificar a hash da senha e gerar o token de acesso reside exclusivamente no `AuthService`, isolando a lógica de segurança do controlador.
* - Utiliza `HttpUtil` para garantir consistência no formato da resposta JSON, independentemente do sucesso ou falha da autenticação.
*
* @return void O método encerra o processamento enviando uma resposta HTTP JSON ao cliente.
* @see AuthService::login() Para os detalhes sobre a regra de autenticação e geração de tokens.
*/
public function login(): void {
try {
$form = HttpUtil::getRequestBody(form_type: 'POST');
if (empty($form)) {
throw new AppException(message: 'Requisição inválida.', code: 400);
}
$authService = ServiceFactory::makeAuthService();
$response = $authService->login(login: SanitizeUtil::string(value: $form['login']), senha: SanitizeUtil::string(value: $form['senha']));
HttpUtil::jsonResponse(
response_code: $response['response_code'] ?? 200,
message: $response['message'] ?? 'Login realizado com sucesso.',
output: $response['output'] ?? null
);
} catch(AppException $e) {
HttpUtil::jsonResponse(
response_code: $e->getCode() ?? 500,
message: $e->getMessage(),
output: $e->getDetails() ? ['errors' => $e->getDetails()] : null
);
}
}
/**
* Endpoint de API para a seleção de contexto de empresa ativa.
*
* Este método gerencia a interface HTTP (POST) para permitir que um usuário
* autenticado alterne sua empresa ativa no sistema multi-empresa. O controlador
* realiza a captura do payload, sanitiza o UUID da empresa e delega a
* complexa lógica de troca de sessão e regeneração de tokens ao `AuthService`.
*
* ---
* ## Fluxo de Execução
* 1. **Captura e Validação:** Extrai o corpo da requisição. Se o payload estiver vazio ou malformado, retorna erro 400 (Bad Request).
* 2. **Sanitização:** Aplica `SanitizeUtil::string()` no `empresa_uuid` para garantir a integridade dos dados recebidos.
* 3. **Processamento de Negócio:** Invoca `AuthService::selectCompany()`, que valida o acesso à empresa, gera um novo JWT e registra a sessão.
* 4. **Normalização de Resposta:**
* - **Sucesso (200 OK):** Retorna o novo token de acesso contendo o contexto da empresa selecionada.
* - **Tratamento de Erros:** Captura `AppException` (ex: empresa inexistente, falta de permissão), retornando uma resposta JSON estruturada com os detalhes do erro para o cliente.
*
* ---
* ## Arquitetura de Seleção de Contexto
*
*
* ---
* ## Observações Técnicas
* - Segue o padrão "Thin Controller", onde a orquestração de infraestrutura (cache, banco de dados, JWT) é encapsulada na camada de serviço.
* - Este endpoint é fundamental para aplicações multi-tenant ou multi-empresa, permitindo que o usuário altere seu "escopo" de atuação sem precisar realizar um re-login completo.
*
* @return void O método encerra o processamento enviando uma resposta HTTP JSON ao cliente.
* @see AuthService::selectCompany() Para os detalhes sobre a regra de negócio aplicada na troca de contexto.
*/
public function selectCompany(): void {
try {
$form = HttpUtil::getRequestBody(form_type: 'POST');
if (empty($form)) {
throw new AppException(message: 'Requisição inválida.', code: 400);
}
$authService = ServiceFactory::makeAuthService();
$response = $authService->selectCompany(empresa_uuid: SanitizeUtil::string(value: $form['empresa_uuid']));
HttpUtil::jsonResponse(
response_code: $response['response_code'] ?? 200,
message: $response['message'] ?? 'Empresa selecionada com sucesso.',
output: $response['output'] ?? null
);
} catch(AppException $e) {
HttpUtil::jsonResponse(
response_code: $e->getCode() ?? 500,
message: $e->getMessage(),
output: $e->getDetails() ? ['errors' => $e->getDetails()] : null
);
}
}
}
@@ -0,0 +1,29 @@
<?php
namespace WorkbloomERP\Module\v0\Auth\Factories;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Module\v0\Auth\Services\AuthService;
use WorkbloomERP\Module\v0\Empresa\Repos\EmpresaRepo;
use WorkbloomERP\Module\v0\Usuario\Repos\UsuarioRepo;
use WorkbloomERP\Module\v0\Usuario\Repos\UsuarioSessionRepo;
use WorkbloomERP\Module\v0\Usuario\Repos\UsuarioEmpresaRepo;
class ServiceFactory {
public static function makeAuthService() {
$db = new DBService();
return new AuthService(
db: $db,
usuarioRepo: new UsuarioRepo(db: $db),
empresaRepo: new EmpresaRepo(db: $db),
usuarioSessionRepo: new UsuarioSessionRepo(db: $db),
usuarioEmpresaRepo: new UsuarioEmpresaRepo(db: $db),
);
}
public static function makeUsuarioSessionRepo() {
$db = new DBService();
return new UsuarioSessionRepo(
db: $db
);
}
}
@@ -0,0 +1,217 @@
<?php
namespace WorkbloomERP\Module\v0\Auth\Middlewares;
use WorkbloomERP\Utils\JwtUtil;
use KrothiumAPI\Utils\HttpUtil;
use WorkbloomERP\Utils\CacheUtil;
use WorkbloomERP\Utils\CypherUtil;
use WorkbloomERP\Exceptions\AppException;
use WorkbloomERP\Module\v0\Auth\Factories\ServiceFactory;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioSessionModel;
class AuthMiddleware {
/**
* Valida se o token JWT da requisição atual corresponde a um estado de "Login Inicial".
*
* Este método atua como um guardião de segurança (Guard) para rotas que exigem
* ações obrigatórias do usuário logo após o primeiro acesso ao sistema, como a
* troca obrigatória de senha ou a configuração de MFA (Autenticação de Dois Fatores).
*
* ---
* ## Fluxo de Execução
* 1. **Extração:** Obtém o token JWT do cabeçalho `Authorization: Bearer`.
* 2. **Validação:** Verifica a integridade e assinatura do token através do `JwtUtil`.
* 3. **Verificação de Claim:** Decodifica o token e confirma se a claim `initial_login` está presente e definida como `true`.
* 4. **Retorno:**
* - Retorna `true` se o token for válido e corresponder ao estado de login inicial.
* - Retorna um `array` estruturado contendo detalhes do erro caso a validação falhe.
*
* ---
* ## Observações Técnicas
* - O método utiliza variáveis de ambiente (`$_ENV`) para configurar o `JwtUtil`, garantindo que as configurações de `secretKey`, `algorithm` e `ttl` estejam sincronizadas com a política de segurança da aplicação.
* - Este método captura internamente instâncias de `AppException`, tratando o fluxo de erro de forma silenciosa para o chamador, retornando um status de erro estruturado em vez de propagar a exceção.
*
* @return true|array{status: string, response_code: int, message: string}
* Retorna `true` em caso de sucesso. Retorna um array com o erro caso o token seja inválido, ausente ou não possua a claim necessária.
*/
public static function isInitialLogin(): true|array {
try {
// Valida o token JWT presente no header Authorization (Bearer Token)
$bearerToken = HttpUtil::getBearerToken();
if (!$bearerToken) {
throw new AppException(message: 'Token de acesso inválido ou ausente.', code: 401);
}
$jwtUtil = new JwtUtil(
secretKey: $_ENV['JWT_SECRET_KEY'], algorithm: $_ENV['JWT_ALGORITHM'],
ttl: $_ENV['JWT_EXPIRATION_TIME'], issuer: $_ENV['JWT_ISSUER']
);
$decodedToken = $jwtUtil->validate(token: $bearerToken);
if (!$decodedToken) {
throw new AppException(message: 'Token de acesso inválido ou ausente.', code: 401);
}
$decodedToken = json_decode(json: json_encode($decodedToken, flags: JSON_THROW_ON_ERROR), associative: true, flags: JSON_THROW_ON_ERROR);
$userData = $decodedToken['user_data'] ?? null;
if (!$userData) {
throw new AppException(message: 'Token de acesso inválido ou ausente.', code: 401);
}
if (!isset($userData['initial_login']) || $userData['initial_login'] !== true) {
throw new AppException(message: 'Token de acesso não corresponde a um login inicial.', code: 403);
}
return true;
} catch(AppException $e) {
return [
'status' => 'error',
'response_code' => $e->getCode() ?? 500,
'message' => $e->getMessage()
];
}
}
/**
* Middleware de autenticação para validar sessões e tokens JWT em requisições protegidas.
*
* Este método atua como um "Gatekeeper" (Guard). Ele implementa uma estratégia híbrida
* de validação (Cache-First, seguida de DB e JWT) para equilibrar performance com
* segurança. Ele verifica a existência do token, a validade da assinatura JWT,
* a revogação da sessão no banco de dados e a integridade do dispositivo (IP/User-Agent).
*
* ---
* ## Fluxo de Execução
* 1. **Extração:** Obtém o token do header `Authorization`. Retorna erro 401 se ausente.
* 2. **Cache-First Check:** Verifica no Redis (`login:main:{token}`). Se encontrado, autoriza imediatamente (retorna `null`), evitando carga no banco de dados.
* 3. **Validação de Persistência:** Consulta o `UsuarioSessionRepo` (via hash do token). Valida revogação (`revoked_at`) e integridade.
* 4. **Validação JWT:** Decodifica e verifica a assinatura/expiração (`exp`) do JWT.
* 5. **Validação de Integridade (Guard):** Invoca `validateSessionData` para garantir que o IP e User-Agent atuais coincidem com a sessão original.
* 6. **Caching:** Se tudo estiver correto, armazena o payload no cache para acelerar futuras requisições e retorna `null` (acesso concedido).
*
* ---
* ## Arquitetura de Validação
*
*
* ---
* ## Observações Técnicas
* - **Interface de Retorno:**
* - `null`: A autenticação foi bem-sucedida; o fluxo de execução pode continuar.
* - `array`: A autenticação falhou; o array contém a resposta de erro estruturada para o cliente.
* - **Segurança:** Utiliza `CypherUtil` para garantir que tokens sensíveis não sejam comparados em texto puro contra o banco de dados.
*
* @return array|null Retorna `null` se o usuário estiver autenticado e autorizado. Retorna um `array` com `response_code`, `message` e `output` em caso de erro.
* @see JwtUtil::validate() Para detalhes sobre a validação da assinatura criptográfica.
* @see self::validateSessionData() Para a verificação de segurança de IP e User-Agent.
*/
public static function handle(): ?array {
try {
// Valida o token JWT presente no header Authorization (Bearer Token)
$bearerToken = HttpUtil::getBearerToken();
if (!$bearerToken) {
throw new AppException(message: 'Token de acesso inválido ou ausente.', code: 401);
}
$jwtUtil = new JwtUtil(
secretKey: $_ENV['JWT_SECRET_KEY'], algorithm: $_ENV['JWT_ALGORITHM'],
ttl: $_ENV['JWT_EXPIRATION_TIME'], issuer: $_ENV['JWT_ISSUER']
);
// Valida o token JWT e decodifica seu payload para extrair os dados do usuário e as informações de sessão.
$decodedToken = $jwtUtil->validate(token: $bearerToken);
if (!$decodedToken) {
throw new AppException(message: 'Token de acesso inválido ou ausente.', code: 401);
}
// Verifica se o token corresponde a um login inicial, bloqueando o acesso a rotas protegidas caso seja verdadeiro.
$decodedToken = json_decode(json: json_encode($decodedToken, flags: JSON_THROW_ON_ERROR), associative: true, flags: JSON_THROW_ON_ERROR);
if ($decodedToken['user_data']['initial_login'] == true) {
throw new AppException(message: 'Acesso negado. O token corresponde a um login inicial e não pode ser usado para acessar esta rota.', code: 403);
}
// Verifica se a sessão principal já está em cache para evitar consultas desnecessárias ao banco de dados e validações repetitivas.
// Se a sessão estiver em cache, considera o token como válido e retorna null para permitir o acesso à rota protegida.
if (CacheUtil::get(key: "login:main:{$bearerToken}")) {
return null;
}
$usuarioSessionRepo = ServiceFactory::makeUsuarioSessionRepo();
// Busca a sessão do usuário no banco de dados utilizando o hash do token para garantir que o token é válido e não foi revogado.
$usuarioSessionModel = $usuarioSessionRepo->findByIdentifier(identifier: 'token_hash', value: CypherUtil::hash(data: $bearerToken));
if (!$usuarioSessionModel || !CypherUtil::verify(data: $bearerToken, hash: $usuarioSessionModel->getTokenHash()) || $usuarioSessionModel->getRevokedAt()) {
throw new AppException(message: 'Sessão inválida ou expirada. Faça login novamente.', code: 401);
}
// Valida o token JWT e decodifica seu payload para extrair os dados do usuário e as informações de sessão.
// Se o token for inválido, expirado ou tiver sua assinatura comprometida, uma exceção é lançada.
$decodedToken = $jwtUtil->validate(token: $bearerToken);
if (!$decodedToken) {
throw new AppException(message: 'Token de acesso inválido ou ausente.', code: 401);
}
$decodedToken = json_decode(json: json_encode($decodedToken, JSON_THROW_ON_ERROR), associative: true, flags: JSON_THROW_ON_ERROR);
$remainingTime = $decodedToken['exp'] - time();
if ($remainingTime <= 0) {
throw new AppException(message: 'Sessão expirada. Faça login novamente.', code: 401);
}
self::validateSessionData(sessionModel: $usuarioSessionModel, tokenData: $decodedToken);
CacheUtil::set(key: "login:main:{$bearerToken}", ttl: $remainingTime, value: $decodedToken);
return null;
} catch (AppException $e) {
return [
'status' => 'error',
'response_code' => $e->getCode() ?: 500,
'message' => $e->getMessage(),
'output' => $e->getDetails() ? ['errors' => $e->getDetails()] : null,
];
}
}
/**
* Valida a integridade e autenticidade da sessão do usuário comparando metadados.
*
* Este método atua como uma barreira de segurança (Guard) que garante que a sessão
* atual está sendo utilizada pelo mesmo dispositivo e rede que a iniciou. Ao comparar
* o IP e o User-Agent da requisição atual com os dados persistidos no banco de dados
* (`UsuarioSessionModel`) e os metadados contidos no token JWT (`$tokenData`),
* protegemos o sistema contra ataques de sequestro de sessão (session hijacking).
*
* ---
* ## Fluxo de Execução
* 1. **Captura:** Extrai o IP (`REMOTE_ADDR`) e o User-Agent (`HTTP_USER_AGENT`) da variável superglobal `$_SERVER`.
* 2. **Comparação (Match):** Utiliza uma expressão `match` para validar quatro pontos de convergência:
* - IP da Sessão (DB) vs. IP Atual.
* - User-Agent da Sessão (DB) vs. User-Agent Atual.
* - IP registrado no Payload do Token vs. IP Atual.
* - User-Agent registrado no Payload do Token vs. User-Agent Atual.
* 3. **Bloqueio:** Caso qualquer divergência seja detectada, a sessão é considerada inválida e uma `AppException` (401 Unauthorized) é disparada.
*
* ---
* ## Observações Técnicas
* - O uso de `match` (PHP 8+) torna o código altamente legível e performático para a verificação booleana múltipla.
* - Este método é fundamental em cenários onde a segurança de acesso exige que o usuário não troque de rede ou navegador enquanto a sessão estiver ativa.
* - Caso a requisição venha através de um proxy, certifique-se de que a configuração de captura de IP (`REMOTE_ADDR`) esteja ajustada corretamente no ambiente.
*
* @param UsuarioSessionModel $sessionModel O modelo contendo os dados da sessão persistidos no banco.
* @param array $tokenData O payload decodificado do token JWT contendo os metadados da sessão.
* @return void
* @throws AppException Caso haja qualquer incompatibilidade nos dados (IP ou User-Agent), invalidando a sessão.
*/
private static function validateSessionData(UsuarioSessionModel $sessionModel, array $tokenData): void {
$ipAddress = $_SERVER['REMOTE_ADDR'] ?? 'unknown';
$userAgent = $_SERVER['HTTP_USER_AGENT'] ?? 'unknown';
$isInvalidSession = match(true) {
$sessionModel->getIpAddress() !== $ipAddress,
$sessionModel->getUserAgent() !== $userAgent,
$tokenData['session_data']['ip_address'] !== $ipAddress,
$tokenData['session_data']['user_agent'] !== $userAgent => true,
default => false
};
if ($isInvalidSession) {
throw new AppException(message: 'Sessão inválida ou expirada. Faça login novamente.', code: 401);
}
}
}
+15
View File
@@ -0,0 +1,15 @@
<?php
use KrothiumAPI\Http\Router;
use WorkbloomERP\Module\v0\Auth\Middlewares\AuthMiddleware;
use WorkbloomERP\Module\v0\Auth\Controllers\AuthController;
Router::group(
prefix: '/auth',
callback: function() {
// Rota de autenticação
Router::post(uri: '/login', handler: [AuthController::class, 'login']);
// Rota para seleção de empresa após login inicial
Router::post(uri: '/select-company', handler: [AuthController::class, 'selectCompany'], middlewares: [[AuthMiddleware::class, 'isInitialLogin']]);
}
);
+289
View File
@@ -0,0 +1,289 @@
<?php
namespace WorkbloomERP\Module\v0\Auth\Services;
use Ramsey\Uuid\Uuid;
use DateTimeImmutable;
use KrothiumAPI\Utils\HttpUtil;
use WorkbloomERP\Utils\JwtUtil;
use WorkbloomERP\Utils\CacheUtil;
use WorkbloomERP\Utils\CypherUtil;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Exceptions\AppException;
use WorkbloomERP\Module\v0\Auth\Utils\AuthUtil;
use WorkbloomERP\Module\v0\Empresa\Repos\EmpresaRepo;
use WorkbloomERP\Module\v0\Usuario\Repos\UsuarioRepo;
use WorkbloomERP\Module\v0\Usuario\Repos\UsuarioSessionRepo;
use WorkbloomERP\Module\v0\Usuario\Repos\UsuarioEmpresaRepo;
use WorkbloomERP\Module\v0\Empresa\Models\EmpresaModel;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioModel;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioSessionModel;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioEmpresaModel;
class AuthService {
public function __construct(
private DBService $db,
private UsuarioRepo $usuarioRepo,
private EmpresaRepo $empresaRepo,
private UsuarioSessionRepo $usuarioSessionRepo,
private UsuarioEmpresaRepo $usuarioEmpresaRepo,
) {}
/**
* Realiza a autenticação de usuário e inicia uma sessão no sistema.
*
* Este método é o coração do processo de autenticação. Ele valida as credenciais
* fornecidas, gera um token JWT (JSON Web Token) para comunicação segura, persiste
* a sessão no banco de dados para fins de auditoria/rastreabilidade e armazena
* o token no cache (Redis) para validação rápida de requests subsequentes.
*
* ---
* ## Fluxo de Execução
* 1. **Validação de Entrada:** Verifica se `login` e `senha` foram fornecidos.
* 2. **Autenticação:** Busca o usuário pelo login e utiliza `password_verify` para validar a hash da senha.
* 3. **Geração de Token:** Instancia o `JwtUtil` com configurações de ambiente e gera um novo token JWT.
* 4. **Persistência de Sessão (Atomicidade):**
* - Insere o registro de sessão na tabela `usuario_session` (contendo dados como IP e User-Agent).
* - O hash do token é armazenado para permitir revogação futura.
* 5. **Caching:** Armazena o token no Redis com a mesma TTL da expiração do token para otimizar a validação de acesso.
*
* ---
* ## Observações Técnicas
* - **Transacionalidade:** Toda a criação de sessão e persistência ocorre dentro de uma transação de banco de dados, garantindo que não existam sessões órfãs caso o cache ou a criação do token falhem.
* - **Segurança:** O token JWT gerado é hasheado (`CypherUtil::hash`) antes de ser salvo no banco para evitar a exposição do token real em caso de vazamento de dados.
* - **Infraestrutura:** Requer acesso funcional a: `JWT_SECRET_KEY`, `JWT_ALGORITHM`, `JWT_EXPIRATION_TIME`, `JWT_ISSUER`.
*
* @param string $login O nome de usuário ou e-mail.
* @param string $senha A senha em texto puro fornecida pelo usuário.
* @return array{response_code: int, message: string, output: array{token: string}} Resposta estruturada contendo o código de sucesso e o token JWT.
* @throws AppException Caso as credenciais sejam inválidas (401), o payload esteja vazio (400) ou falhas críticas ocorram na geração de token/sessão (500).
*/
public function login(string $login, string $senha): array {
try {
// Validação básica de entrada
if (empty($login) || empty($senha)) {
throw new AppException(message: 'Login e senha são obrigatórios.', code: 400);
}
// Inicia uma transação para garantir a atomicidade das operações de autenticação
return $this->db->transaction(
callback: function() use ($login, $senha) {
$usuarioModel = $this->usuarioRepo->findByLogin(login: $login);
if (!$usuarioModel || !password_verify(password: $senha, hash: $usuarioModel->getSenhaHash())) {
throw new AppException(message: 'Login ou senha inválidos.', code: 401);
}
$payload = [
'user_data' => [
'initial_login' => true,
'uuid' => $usuarioModel->getUuid(),
'nome_usuario' => $usuarioModel->getNomeCompleto(),
'email' => $usuarioModel->getEmail(),
],
'exp' => (new DateTimeImmutable())->modify("+5 minutes")->getTimestamp(),
];
$jwtUtil = new JwtUtil(
secretKey: $_ENV['JWT_SECRET_KEY'], algorithm: $_ENV['JWT_ALGORITHM'],
ttl: $_ENV['JWT_EXPIRATION_TIME'], issuer: $_ENV['JWT_ISSUER']
);
$jwtToken = $jwtUtil->generate(payload: $payload);
if (!$jwtToken) {
throw new AppException(message: 'Erro ao gerar token de acesso.', code: 500);
}
// Armazena o token JWT no cache Redis com uma TTL de 10 minutos (600 segundos)
CacheUtil::set(key: "login:init:{$jwtToken}", ttl: 600, value: $payload);
return [
'response_code' => 200,
'message' => 'Login realizado com sucesso.',
'output' => [
'tmp_token' => $jwtToken,
'empresas' => $this->getEmpresasByUsuarioId($usuarioModel->getId())
]
];
}
);
} catch(AppException $e) {
throw $e;
}
}
/**
* Recupera e organiza a lista de empresas vinculadas a um determinado usuário.
*
* Este método realiza o mapeamento entre o ID do usuário e suas respectivas
* empresas. Ele filtra as associações através do repositório `usuarioEmpresaRepo`,
* hidrata os objetos `EmpresaModel` e organiza os dados em uma estrutura
* associativa agrupada pelo nome empresarial.
*
* ---
* ## Fluxo de Execução
* 1. **Busca:** Recupera todos os registros de associação `usuario_id` -> `empresa_id`.
* 2. **Hidratação:** Itera sobre as associações, buscando o `EmpresaModel` completo para cada ID encontrado.
* 3. **Estruturação:** Monta um array multidimensional onde as chaves são os nomes das empresas e os valores são arrays contendo metadados (UUID, nome, CNPJ, tipo formatado).
* 4. **Tratamento de Dados:** Normaliza o campo `tipo` (MATRIZ/FILIAL) para o formato "Capitalizado" (ex: Matriz).
*
* ---
* ## Observações Técnicas
* - Caso o usuário não possua empresas vinculadas ou o repositório retorne vazio, o método retorna um array vazio.
* - Ignora silenciosamente qualquer `empresa_id` que não resulte em um `EmpresaModel` válido (falha de integridade referencial).
* - O agrupamento pelo nome empresarial facilita o consumo da estrutura pelo frontend.
*
* @param int $usuario_id O identificador numérico interno do usuário.
* @return array<string, array<int, array{uuid: string, nome_empresarial: string, nome_fantasia: string, cnpj: string, tipo: string}>> Array estruturado de empresas.
* @throws AppException Caso ocorra falha de acesso aos repositórios.
*/
private function getEmpresasByUsuarioId(int $usuario_id): array {
try {
$empresas = $this->usuarioEmpresaRepo->findAllByUsuarioId(usuario_id: $usuario_id);
if (empty($empresas)) {
return ['Nenhuma empresa vinculada ao usuário.'];
}
$preparedEmpresas = [];
foreach($empresas as $i => $empresa) {
$empresaModel = $this->empresaRepo->findByIdentifier(identifier: 'id', value: $empresa['empresa_id']);
if (!$empresaModel) {
continue;
}
$preparedEmpresas[$empresaModel->getNomeEmpresarial()][] = [
'uuid' => $empresaModel->getUuid(),
'nome_empresarial' => $empresaModel->getNomeEmpresarial(),
'nome_fantasia' => $empresaModel->getNomeFantasia(),
'cnpj' => $empresaModel->getDocumentCnpj(),
'tipo' => ucfirst(string: strtolower(string: $empresaModel->getTipo()))
];
}
return $preparedEmpresas;
} catch(AppException $e) {
throw $e;
}
}
/**
* Seleciona e define a empresa ativa para o contexto da sessão do usuário.
*
* Este método é utilizado em sistemas multi-empresa onde o usuário precisa alternar
* entre diferentes entidades (empresas) sem realizar um novo login completo.
* O método valida se o usuário possui vínculo com a empresa solicitada, gera
* um novo token JWT com o contexto da empresa atualizado e registra essa nova
* sessão no banco de dados e no cache (Redis).
*
* ---
* ## Fluxo de Execução
* 1. **Verificação de Sessão:** Garante que o usuário já possui um login ativo (leitura via `AuthUtil`).
* 2. **Validação de Acesso:** Verifica se o usuário solicitado possui associação ativa com a empresa (`usuarioEmpresaRepo`).
* 3. **Contextualização:** Monta um novo payload contendo os dados do usuário e os dados da empresa selecionada.
* 4. **Token Rotation:** Gera um novo token JWT com os novos dados de contexto e define um tempo de expiração (60 minutos).
* 5. **Persistência:** Registra a nova sessão no banco de dados (`usuario_session`) e atualiza o estado no cache (`CacheUtil`).
*
* ---
* ## Observações Técnicas
* - **Token Rotation:** O método invalida implicitamente o contexto anterior ao gerar um novo token baseado na nova seleção.
* - **Atomicidade:** A operação ocorre dentro de uma transação, garantindo que se o token não for gerado ou a sessão não for criada, nada é alterado.
* - **Segurança:** O hash do token gerado é armazenado para permitir rastreabilidade e segurança.
*
* @param string $empresa_uuid O Identificador Único Universal da empresa que o usuário deseja selecionar.
* @return array{response_code: int, message: string, output: array{token: string}} Resposta estruturada contendo o novo token de acesso com o contexto atualizado.
* @throws AppException Caso a sessão seja inválida, a empresa não exista, o usuário não tenha permissão de acesso à empresa ou falhas de persistência ocorram.
*/
public function selectCompany(string $empresa_uuid) {
try {
return $this->db->transaction(
callback: function() use ($empresa_uuid) {
if (empty($empresa_uuid)) {
throw new AppException(message: 'UUID da empresa é obrigatório.', code: 400);
}
// Lê os dados da sessão atual para validar a existência de um login prévio e obter o UUID do usuário
$sessionData = AuthUtil::readInitSession();
if (!$sessionData || !isset($sessionData['user_data']['uuid'])) {
throw new AppException(message: 'Sessão de usuário não encontrada. Faça login novamente.', code: 401);
}
// Valida se o usuário tem acesso à empresa selecionada
$usuarioModel = $this->usuarioRepo->findByIdentifier(identifier: 'uuid', value: $sessionData['user_data']['uuid']);
if (!$usuarioModel) {
throw new AppException(message: 'Ocorreu um erro ao selecionar a empresa. Tente novamente mais tarde.', code: 404);
}
// Valida se a empresa existe e se o usuário tem associação com ela
$empresaModel = $this->empresaRepo->findByIdentifier(identifier: 'uuid', value: $empresa_uuid);
if (!$empresaModel || !$this->usuarioEmpresaRepo->checkAssociationByUsuarioIdAndEmpresaId(usuario_id: $usuarioModel->getId(), empresa_id: $empresaModel->getId())) {
throw new AppException(message: 'Empresa não encontrada.', code: 404);
}
// Monta o payload da sessão principal, incluindo os dados do usuário e da empresa selecionada
$payload = [
'user_data' => [
'initial_login' => false,
'uuid' => $usuarioModel->getUuid(),
'nome_completo' => $usuarioModel->getNomeCompleto(),
'nome_usuario' => $usuarioModel->getNomeUsuario(),
'email' => $usuarioModel->getEmail(),
'is_root' => $usuarioModel->getIsRoot() ? true : false
],
'empresa_data' => [
'uuid' => $empresaModel->getUuid(),
'nome_empresarial' => $empresaModel->getNomeEmpresarial(),
'nome_fantasia' => $empresaModel->getNomeFantasia(),
'cnpj' => $empresaModel->getDocumentCnpj(),
'tipo' => ucfirst(string: strtolower(string: $empresaModel->getTipo()))
],
'session_data' => [
'ip_address' => $_SERVER['REMOTE_ADDR'] ?? 'unknown',
'user_agent' => $_SERVER['HTTP_USER_AGENT'] ?? 'unknown'
],
'iat' => (new DateTimeImmutable())->getTimestamp(),
'exp' => (new DateTimeImmutable())->modify("+{$_ENV['JWT_EXPIRATION_TIME']} seconds")->getTimestamp(),
];
$jwtUtil = new JwtUtil(
secretKey: $_ENV['JWT_SECRET_KEY'], algorithm: $_ENV['JWT_ALGORITHM'],
ttl: $_ENV['JWT_EXPIRATION_TIME'], issuer: $_ENV['JWT_ISSUER']
);
$jwtToken = $jwtUtil->generate(payload: $payload);
if (!$jwtToken) {
throw new AppException(message: 'Erro ao gerar token de acesso.', code: 500);
}
$usuarioSessionModel = $this->usuarioSessionRepo->insert(
new UsuarioSessionModel(
uuid: Uuid::uuid7()->toString(),
usuario_id: $usuarioModel->getId(),
token_hash: CypherUtil::hash(data: $jwtToken),
ip_address: $payload['session_data']['ip_address'],
user_agent: $payload['session_data']['user_agent'],
created_at: new DateTimeImmutable()
)
);
if (!$usuarioSessionModel) {
throw new AppException(message: 'Erro ao criar sessão de usuário.', code: 500);
}
if (!CacheUtil::set(key: "login:main:{$jwtToken}", ttl: $_ENV['JWT_EXPIRATION_TIME'], value: $payload)) {
throw new AppException(message: 'Erro ao armazenar sessão no cache.', code: 500);
}
$beforeToken = HttpUtil::getBearerToken();
if (!CacheUtil::delete(keys: ["login:init:{$beforeToken}"])) {
throw new AppException(message: 'Erro ao limpar sessão temporária do cache.', code: 500);
}
return [
'response_code' => 200,
'message' => 'Empresa selecionada com sucesso.',
'output' => [
'token' => $jwtToken
]
];
}
);
} catch(AppException $e) {
throw $e;
}
}
}
+83
View File
@@ -0,0 +1,83 @@
<?php
namespace WorkbloomERP\Module\v0\Auth\Utils;
use KrothiumAPI\Utils\HttpUtil;
use WorkbloomERP\Utils\JwtUtil;
use WorkbloomERP\Utils\CacheUtil;
use WorkbloomERP\Utils\CryptoUtil;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioModel;
use WorkbloomERP\Module\v0\Auth\Factories\ServiceFactory;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioSessionModel;
class AuthUtil {
/**
* Recupera os dados de uma sessão de "Login Inicial" a partir do cache.
*
* Este método é utilizado para verificar se existe uma sessão ativa vinculada ao
* token de acesso atual no fluxo de "Primeiro Acesso" (Initial Login). Ele permite
* identificar rapidamente se o usuário está em um estado que exige ações
* obrigatórias, como a alteração de senha inicial ou configuração de MFA,
* sem a necessidade de consultar o banco de dados.
*
* ---
* ## Fluxo de Execução
* 1. **Identificação:** Extrai o token Bearer da requisição atual (`HttpUtil::getBearerToken()`).
* 2. **Busca em Cache:** Consulta o serviço de cache (Redis) utilizando a chave prefixada `login:init:{token}`.
* 3. **Retorno:** Devolve os dados da sessão (array) caso o registro exista e esteja ativo; retorna `null` caso a sessão não exista ou tenha expirado.
*
* ---
* ## Observações Técnicas
* - **Performance:** Este método é altamente performático por consultar exclusivamente a camada de cache, sendo ideal para middlewares de validação de fluxo de login.
* - **Segurança:** A chave de busca utiliza o próprio token Bearer, garantindo que o acesso aos dados da sessão inicial seja restrito ao detentor do token correspondente.
* - **Integração:** Este método deve ser utilizado em conjunto com o fluxo de autenticação que persiste os dados do login inicial no cache.
*
* @return array|null Retorna o payload da sessão de login inicial se encontrada; retorna `null` se a sessão não existir no cache.
*/
public static function readInitSession(): ?array {
// Verifica se existe um login inicial
$bearerToken = HttpUtil::getBearerToken();
$initSession = CacheUtil::get(key: "login:init:{$bearerToken}");
if ($initSession) {
return $initSession;
}
return null;
}
/**
* Recupera a sessão principal do usuário ou uma chave específica do payload armazenado em cache.
*
* Este método centraliza o acesso aos dados de sessão ativos. Ele utiliza o
* token Bearer da requisição atual para localizar o payload no cache (Redis) e,
* caso uma chave específica seja fornecida, retorna apenas o subconjunto de dados
* solicitado, otimizando o acesso às informações de contexto.
*
* ---
* ## Fluxo de Execução
* 1. **Identificação:** Obtém o token de acesso (Bearer Token) da requisição atual via `HttpUtil`.
* 2. **Busca:** Consulta o cache usando a chave padronizada `login:main:{token}`.
* 3. **Filtragem:**
* - Se uma `$key` for fornecida e existir no payload, retorna apenas o valor correspondente (ex: `user_data`).
* - Se nenhuma chave for fornecida ou a chave não existir, retorna o payload completo.
* 4. **Retorno:** Devolve `null` caso a sessão não esteja em cache (sessão inexistente ou expirada).
*
* ---
* ## Observações Técnicas
* - **Performance:** A consulta ao cache é direta e rápida. O método evita redundância ao permitir filtrar os dados necessários logo na leitura.
* - **Flexibilidade:** A inclusão do parâmetro `$key` permite que outros serviços solicitem apenas o bloco de dados que precisam (ex: dados da empresa ou do usuário), reduzindo o processamento desnecessário de arrays grandes.
*
* @param string $key Chave específica a ser extraída do payload da sessão (opcional; se vazio, retorna o payload completo).
* @return array|null Retorna o payload da sessão ou um subconjunto dele; retorna `null` se a sessão não existir no cache.
*/
public static function readSession(string $key): ?array {
$bearerToken = HttpUtil::getBearerToken();
$result = CacheUtil::get(key: "login:main:{$bearerToken}");
if (!$result) {return null;}
if ($key && isset($result[$key])) {
return $result[$key];
}
return $result;
}
}
@@ -0,0 +1,22 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Constants;
class ContribuinteICMSConst {
public const CONTATO_CONTRIBUINTE_ICMS = [
1 => 'Contribuinte ICMS',
2 => 'Contribuinte Isento de Inscrição no Cadastro de Contribuintes',
9 => 'Não Contribuinte, que pode ou não possuir Inscrição Estadual no Cadastro de Contribuintes'
];
public static function getAll(): array {
return self::CONTATO_CONTRIBUINTE_ICMS;
}
public static function isValid(string $value): bool {
return array_key_exists(key: $value, array: self::CONTATO_CONTRIBUINTE_ICMS);
}
public static function getDescription(string $key): ?string {
return self::CONTATO_CONTRIBUINTE_ICMS[$key] ?? null;
}
}
@@ -0,0 +1,23 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Constants;
class NFSeConsumoIbsCbsConst {
public const CONTATO_USO_CONSUMO_IBS_CBS = [
'1' => 'Sim (Consumidor Final)',
'2' => 'Não informar',
'0' => 'Não (Operação B2B)',
];
public static function getAll(): array {
return self::CONTATO_USO_CONSUMO_IBS_CBS;
}
public static function getDescription(string|int|null $key): string {
return self::CONTATO_USO_CONSUMO_IBS_CBS[$key] ?? 'Valor desconhecido';
}
public static function isValid(string|int|null $key): bool {
return array_key_exists(key: $key, array: self::CONTATO_USO_CONSUMO_IBS_CBS);
}
}
@@ -0,0 +1,23 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Constants;
class OrgaoPublicoConst {
public const CONTATO_ORGAO_PUBLICO = [
'NAO' => 'Não',
'FEDERAL' => 'Órgão Público Federal',
'ESTADUAL' => 'Órgão Público Estadual',
'MUNICIPAL' => 'Órgão Público Municipal'
];
public static function getAll(): array {
return self::CONTATO_ORGAO_PUBLICO;
}
public static function isValid(string $value): bool {
return array_key_exists(key: $value, array: self::CONTATO_ORGAO_PUBLICO);
}
public static function getDescription(string $key): ?string {
return self::CONTATO_ORGAO_PUBLICO[$key] ?? null;
}
}
@@ -0,0 +1,22 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Constants;
class PersonalidadeConst {
public const CONTATO_PERSONALIDADE = [
'PF' => 'Pessoa Física',
'PJ' => 'Pessoa Jurídica',
'EX' => 'Estrangeiro'
];
public static function getAll(): array {
return self::CONTATO_PERSONALIDADE;
}
public static function isValid(string $value): bool {
return array_key_exists(key: $value, array: self::CONTATO_PERSONALIDADE);
}
public static function getDescription(string $key): ?string {
return self::CONTATO_PERSONALIDADE[$key] ?? null;
}
}
@@ -0,0 +1,21 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Constants;
class TipoConst {
public const CONTATO_TIPO = [
'CLIENTE' => 'Cliente',
'FORNECEDOR' => 'Fornecedor'
];
public static function getAll(): array {
return self::CONTATO_TIPO;
}
public static function isValid(string $value): bool {
return array_key_exists(key: $value, array: self::CONTATO_TIPO);
}
public static function getDescription(string $key): ?string {
return self::CONTATO_TIPO[$key] ?? null;
}
}
@@ -0,0 +1,107 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Controllers;
use KrothiumAPI\Utils\HttpUtil;
use WorkbloomERP\Utils\SanitizeUtil;
use WorkbloomERP\Exceptions\AppException;
use WorkbloomERP\Module\v0\Contato\DTOs\ContatoCreateDTO;
use WorkbloomERP\Module\v0\Contato\Factories\ContatoServiceFactory;
class ContatoController {
public function createOptions(): void {
try {
$contatoService = ContatoServiceFactory::makeContatoService();
$options = $contatoService->createOptions();
HttpUtil::jsonResponse(
response_code: 200,
message: 'Opções para criação de contato.',
output: ['options' => $options]
);
} catch (AppException $e) {
HttpUtil::jsonResponse(
response_code: $e->getCode() ?? 500,
message: $e->getMessage() ?? 'Ocorreu um erro ao obter as opções de contato.',
output: $e->getDetails() ? ['errors' => $e->getDetails()] : null
);
}
}
/**
* Endpoint de API para a criação de um novo **Contato**.
*
* Este controlador atua como o ponto de entrada para o registro de novos contatos
* (parceiros de negócio, fornecedores ou clientes) no sistema. O método é
* responsável por extrair os dados da requisição HTTP (POST), aplicar a sanitização
* necessária para garantir a integridade dos dados e delegar a criação à camada de
* serviço utilizando um `ContatoCreateDTO`.
*
* ---
* ## Fluxo de Execução
* 1. **Captura e Validação:** Obtém o corpo da requisição (JSON/Form). Valida a presença de dados, retornando erro 422 caso esteja vazio.
* 2. **Normalização (DTO):** Instancia um `ContatoCreateDTO` populando-o com os dados sanitizados através de `SanitizeUtil`.
* 3. **Orquestração de Serviço:** Invoca `ContatoService::create()` para validar regras de negócio, persistência e possíveis integrações.
* 4. **Resposta:** Retorna um JSON estruturado com o status da operação (201 para sucesso ou status de erro correspondente).
*
* ---
* ## Observações Técnicas
* - O controlador segue o padrão de "Thin Controller", garantindo que as regras de validação complexas e a persistência de dados ocorram na camada de serviço.
* - Todos os campos de entrada são submetidos a métodos de `SanitizeUtil` para prevenir injeções e garantir o tipo esperado antes de atingir o DTO.
* - Utiliza `ContatoServiceFactory` para a injeção de dependência do serviço.
*
* @return void O método encerra o processamento enviando uma resposta HTTP JSON ao cliente.
* @see ContatoCreateDTO Para a estrutura de dados esperada para a criação.
* @see ContatoService::create() Para a lógica de negócio de persistência do contato.
*/
public function create(): void {
try {
$form = HttpUtil::getRequestBody(form_type: 'POST');
if (empty($form)) {
throw new AppException(message: 'Requisição inválida.', code: 422);
}
$contaoService = ContatoServiceFactory::makeContatoService();
$response = $contaoService->create(
contatoCreateDTO: new ContatoCreateDTO(
is_active: SanitizeUtil::boolean(value: $form['is_active'] ?? null),
tipo: SanitizeUtil::string(value: $form['tipo'] ?? null),
nome_empresarial: SanitizeUtil::string(value: $form['nome_empresarial'] ?? null),
nome_fantasia: SanitizeUtil::string(value: $form['nome_fantasia'] ?? null),
personalidade: SanitizeUtil::string(value: $form['personalidade'] ?? null),
document_cpf: SanitizeUtil::string(value: $form['document_cpf'] ?? null),
document_cnpj: SanitizeUtil::string(value: $form['document_cnpj'] ?? null),
regime_tributario: SanitizeUtil::int(value: $form['regime_tributario'] ?? null),
contribuinte_icms: SanitizeUtil::int(value: $form['contribuinte_icms'] ?? null),
orgao_publico: SanitizeUtil::string(value: $form['orgao_publico'] ?? null),
document_ie: SanitizeUtil::string(value: $form['document_ie'] ?? null),
document_im: SanitizeUtil::string(value: $form['document_im'] ?? null),
document_is: SanitizeUtil::string(value: $form['document_is'] ?? null),
end_cep: SanitizeUtil::string(value: $form['end_cep'] ?? null),
end_ibge: SanitizeUtil::string(value: $form['end_ibge'] ?? null),
end_logradouro: SanitizeUtil::string(value: $form['end_logradouro'] ?? null),
end_numero: SanitizeUtil::string(value: $form['end_numero'] ?? null),
end_complemento: SanitizeUtil::string(value: $form['end_complemento'] ?? null),
end_bairro: SanitizeUtil::string(value: $form['end_bairro'] ?? null),
end_cidade: SanitizeUtil::string(value: $form['end_cidade'] ?? null),
end_uf: SanitizeUtil::string(value: $form['end_uf'] ?? null),
info_email: SanitizeUtil::string(value: $form['info_email'] ?? null),
info_email_nfe: SanitizeUtil::string(value: $form['info_email_nfe'] ?? null),
info_observacao: SanitizeUtil::string(value: $form['info_observacao'] ?? null),
info_telefone: SanitizeUtil::string(value: $form['info_telefone'] ?? null),
info_uso_consumo_ibs_cbs: SanitizeUtil::string(value: $form['info_uso_consumo_ibs_cbs'] ?? null),
)
);
HttpUtil::jsonResponse(
response_code: $response['response_code'] ?? 201,
message: $response['message'] ?? 'Contato criado com sucesso.',
output: $response['output'] ?? null
);
} catch (AppException $e) {
HttpUtil::jsonResponse(
response_code: $e->getCode() ?? 500,
message: $e->getMessage() ?? 'Ocorreu um erro ao criar o contato.',
output: $e->getDetails() ? ['errors' => $e->getDetails()] : null
);
}
}
}
@@ -0,0 +1,246 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\DTOs;
class ContatoCreateDTO {
public function __construct(
private ?int $is_active = null,
private ?string $tipo = null,
private ?string $nome_empresarial = null,
private ?string $nome_fantasia = null,
private ?string $personalidade = null,
private ?string $document_cpf = null,
private ?string $document_cnpj = null,
private ?int $regime_tributario = null,
private ?int $contribuinte_icms = null,
private ?string $orgao_publico = null,
private ?string $document_ie = null,
private ?string $document_im = null,
private ?string $document_is = null,
private ?string $end_cep = null,
private ?string $end_ibge = null,
private ?string $end_logradouro = null,
private ?string $end_numero = null,
private ?string $end_complemento = null,
private ?string $end_bairro = null,
private ?string $end_cidade = null,
private ?string $end_uf = null,
private ?string $info_email = null,
private ?string $info_email_nfe = null,
private ?string $info_observacao = null,
private ?string $info_telefone = null,
private ?string $info_uso_consumo_ibs_cbs = null,
) {}
public function toArray(): array {
return [
'is_active' => $this->getIsActive(),
'tipo' => $this->getTipo(),
'nome_empresarial' => $this->getNomeEmpresarial(),
'nome_fantasia' => $this->getNomeFantasia(),
'personalidade' => $this->getPersonalidade(),
'document_cpf' => $this->getDocumentCpf(),
'document_cnpj' => $this->getDocumentCnpj(),
'regime_tributario' => $this->getRegimeTributario(),
'contribuinte_icms' => $this->getContribuinteIcms(),
'orgao_publico' => $this->getOrgaoPublico(),
'document_ie' => $this->getDocumentIe(),
'document_im' => $this->getDocumentIm(),
'document_is' => $this->getDocumentIs(),
'end_cep' => $this->getEndCep(),
'end_ibge' => $this->getEndIbge(),
'end_logradouro' => $this->getEndLogradouro(),
'end_numero' => $this->getEndNumero(),
'end_complemento' => $this->getEndComplemento(),
'end_bairro' => $this->getEndBairro(),
'end_cidade' => $this->getEndCidade(),
'end_uf' => $this->getEndUf(),
'info_email' => $this->getInfoEmail(),
'info_email_nfe' => $this->getInfoEmailNfe(),
'info_observacao' => $this->getInfoObservacao(),
'info_telefone' => $this->getInfoTelefone(),
'info_uso_consumo_ibs_cbs' => $this->getInfoUsoConsumoIbsCbs(),
];
}
public function setIsActive(?int $is_active): void {
$this->is_active = $is_active;
}
public function getIsActive(): ?int {
return $this->is_active;
}
public function setTipo(?string $tipo): void {
$this->tipo = $tipo;
}
public function getTipo(): ?string {
return $this->tipo;
}
public function setNomeEmpresarial(?string $nome_empresarial): void {
$this->nome_empresarial = $nome_empresarial;
}
public function getNomeEmpresarial(): ?string {
return $this->nome_empresarial;
}
public function setNomeFantasia(?string $nome_fantasia): void {
$this->nome_fantasia = $nome_fantasia;
}
public function getNomeFantasia(): ?string {
return $this->nome_fantasia;
}
public function setPersonalidade(?string $personalidade): void {
$this->personalidade = $personalidade;
}
public function getPersonalidade(): ?string {
return $this->personalidade;
}
public function setDocumentCpf(?string $document_cpf): void {
$this->document_cpf = $document_cpf;
}
public function getDocumentCpf(): ?string {
return $this->document_cpf;
}
public function setDocumentCnpj(?string $document_cnpj): void {
$this->document_cnpj = $document_cnpj;
}
public function getDocumentCnpj(): ?string {
return $this->document_cnpj;
}
public function setRegimeTributario(?int $regime_tributario): void {
$this->regime_tributario = $regime_tributario;
}
public function getRegimeTributario(): ?int {
return $this->regime_tributario;
}
public function setContribuinteIcms(?int $contribuinte_icms): void {
$this->contribuinte_icms = $contribuinte_icms;
}
public function getContribuinteIcms(): ?int {
return $this->contribuinte_icms;
}
public function setOrgaoPublico(?string $orgao_publico): void {
$this->orgao_publico = $orgao_publico;
}
public function getOrgaoPublico(): ?string {
return $this->orgao_publico;
}
public function setDocumentIe(?string $document_ie): void {
$this->document_ie = $document_ie;
}
public function getDocumentIe(): ?string {
return $this->document_ie;
}
public function setDocumentIm(?string $document_im): void {
$this->document_im = $document_im;
}
public function getDocumentIm(): ?string {
return $this->document_im;
}
public function setDocumentIs(?string $document_is): void {
$this->document_is = $document_is;
}
public function getDocumentIs(): ?string {
return $this->document_is;
}
public function setEndCep(?string $end_cep): void {
$this->end_cep = $end_cep;
}
public function getEndCep(): ?string {
return $this->end_cep;
}
public function setEndIbge(?string $end_ibge): void {
$this->end_ibge = $end_ibge;
}
public function getEndIbge(): ?string {
return $this->end_ibge;
}
public function setEndLogradouro(?string $end_logradouro): void {
$this->end_logradouro = $end_logradouro;
}
public function getEndLogradouro(): ?string {
return $this->end_logradouro;
}
public function setEndNumero(?string $end_numero): void {
$this->end_numero = $end_numero;
}
public function getEndNumero(): ?string {
return $this->end_numero;
}
public function setEndComplemento(?string $end_complemento): void {
$this->end_complemento = $end_complemento;
}
public function getEndComplemento(): ?string {
return $this->end_complemento;
}
public function setEndBairro(?string $end_bairro): void {
$this->end_bairro = $end_bairro;
}
public function getEndBairro(): ?string {
return $this->end_bairro;
}
public function setEndCidade(?string $end_cidade): void {
$this->end_cidade = $end_cidade;
}
public function getEndCidade(): ?string {
return $this->end_cidade;
}
public function setEndUf(?string $end_uf): void {
$this->end_uf = $end_uf;
}
public function getEndUf(): ?string {
return $this->end_uf;
}
public function setInfoEmail(?string $info_email): void {
$this->info_email = $info_email;
}
public function getInfoEmail(): ?string {
return $this->info_email;
}
public function setInfoEmailNfe(?string $info_email_nfe): void {
$this->info_email_nfe = $info_email_nfe;
}
public function getInfoEmailNfe(): ?string {
return $this->info_email_nfe;
}
public function setInfoObservacao(?string $info_observacao): void {
$this->info_observacao = $info_observacao;
}
public function getInfoObservacao(): ?string {
return $this->info_observacao;
}
public function setInfoTelefone(?string $info_telefone): void {
$this->info_telefone = $info_telefone;
}
public function getInfoTelefone(): ?string {
return $this->info_telefone;
}
public function setInfoUsoConsumoIbsCbs(?string $info_uso_consumo_ibs_cbs): void {
$this->info_uso_consumo_ibs_cbs = $info_uso_consumo_ibs_cbs;
}
public function getInfoUsoConsumoIbsCbs(): ?string {
return $this->info_uso_consumo_ibs_cbs;
}
}
@@ -0,0 +1,173 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Factories;
use DateTimeImmutable;
use Ramsey\Uuid\Uuid;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Utils\ValidateUtil;
use WorkbloomERP\Exceptions\AppException;
use WorkbloomERP\Module\v0\Empresa\Repos\EmpresaRepo;
use WorkbloomERP\Module\v0\Contato\Repos\ContatoRepo;
use WorkbloomERP\Module\v0\Contato\Models\ContatoModel;
use WorkbloomERP\Module\v0\Empresa\Models\EmpresaModel;
use WorkbloomERP\Module\v0\Contato\DTOs\ContatoCreateDTO;
// Constantes para opções de contato
use WorkbloomERP\Constants\SpedCRTConst;
use WorkbloomERP\Constants\BrasilUfsConst;
use WorkbloomERP\Constants\SpedPaisesConst;
use WorkbloomERP\Module\v0\Contato\Constants\OrgaoPublicoConst;
use WorkbloomERP\Module\v0\Contato\Constants\PersonalidadeConst;
use WorkbloomERP\Module\v0\Contato\Constants\ContribuinteICMSConst;
use WorkbloomERP\Module\v0\Contato\Constants\NFSeConsumoIbsCbsConst;
use WorkbloomERP\Module\v0\Contato\Constants\TipoConst;
class ContatoFactory {
public function __construct(private EmpresaModel $empresaModel, private ContatoCreateDTO $contatoCreateDTO, private ContatoRepo $contatoRepo, private EmpresaRepo $empresaRepo, private DBService $db) {
if (!$empresaModel) {
throw new AppException(message: 'Empresa não encontrada.', code: 404);
}
if (!$contatoCreateDTO) {
throw new AppException(message: 'Dados do contato inválidos.', code: 400);
}
}
public function create(ContatoCreateDTO $contatoCreateDTO, EmpresaModel $empresaModel): ?ContatoModel {
try {
// Valida os campos comuns a ambos os tipos de contato (PF e PJ)
$this->validateCommonFields(contatoCreateDTO: $contatoCreateDTO);
// Cria o contato com base na personalidade (PF ou PJ)
$contatoModel = match($contatoCreateDTO->getPersonalidade()) {
'PF' => $this->createPF(contatoCreateDTO: $contatoCreateDTO, empresaModel: $empresaModel),
'PJ' => $this->createPJ(contatoCreateDTO: $contatoCreateDTO, empresaModel: $empresaModel)
};
return $contatoModel;
} catch(AppException $e) {
throw $e;
}
}
private function validateCommonFields(ContatoCreateDTO $contatoCreateDTO): void {
$errors = [];
$commonFields = [
'is_active',
'tipo',
'nome_empresarial',
'personalidade',
'regime_tributario',
'contribuinte_icms',
'orgao_publico',
'end_cep',
'end_logradouro',
'end_numero',
'end_bairro',
'end_cidade',
'end_uf',
'info_email',
'info_email_nfe',
'info_telefone',
'info_uso_consumo_ibs_cbs',
];
foreach ($commonFields as $field) {
if (!array_key_exists(key: $field, array: $contatoCreateDTO->toArray()) || $contatoCreateDTO->toArray()[$field] === '') {
$errors[$field] = 'Campo obrigatório.';
}
}
// Verifica se o campo 'tipo' tem um valor válido
$tipo = $contatoCreateDTO->getTipo() ? strtoupper(string: $contatoCreateDTO->getTipo()) : null;
if ($tipo && !TipoConst::isValid(value: $tipo)) {
$errors['tipo'] = "Valor inválido.";
}
// Verifica se o campo 'personalidade' tem um valor válido
$personalidade = $contatoCreateDTO->getPersonalidade() ? strtoupper(string: $contatoCreateDTO->getPersonalidade()) : null;
if ($personalidade && !PersonalidadeConst::isValid(value: $personalidade)) {
$errors['personalidade'] = "Valor inválido.";
}
// Valida o regime tributário
if (!in_array($contatoCreateDTO->getRegimeTributario(), [1, 2, 3, 4, null], true)) {
$errors['regime_tributario'] = "Valor inválido.";
}
if ($errors) {
throw new AppException(message: 'Dados de contato inválidos.',code: 422, details: $errors);
}
}
private function createPF(ContatoCreateDTO $contatoCreateDTO, EmpresaModel $empresaModel): ContatoModel {
try {
// Valida os campos específicos para pessoa física
$this->validatePFFields(contatoCreateDTO: $contatoCreateDTO, empresaModel: $empresaModel);
$contatoModel = new ContatoModel();
$contatoModel->setUuid(Uuid::uuid7()->toString());
$contatoModel->setEmpresaId($empresaModel->getId());
$contatoModel->setIsActive($contatoCreateDTO->getIsActive());
$contatoModel->setTipo($contatoCreateDTO->getTipo());
$contatoModel->setNomeEmpresarial($contatoCreateDTO->getNomeEmpresarial());
$contatoModel->setPersonalidade($contatoCreateDTO->getPersonalidade());
$contatoModel->setRegimeTributario($contatoCreateDTO->getRegimeTributario());
$contatoModel->setContribuinteIcms($contatoCreateDTO->getContribuinteIcms());
$contatoModel->setOrgaoPublico($contatoCreateDTO->getOrgaoPublico());
$contatoModel->setEndCep($contatoCreateDTO->getEndCep());
$contatoModel->setEndIbge($contatoCreateDTO->getEndIbge());
$contatoModel->setEndLogradouro($contatoCreateDTO->getEndLogradouro());
$contatoModel->setEndNumero($contatoCreateDTO->getEndNumero());
$contatoModel->setEndBairro($contatoCreateDTO->getEndBairro());
$contatoModel->setEndCidade($contatoCreateDTO->getEndCidade());
$contatoModel->setEndUf($contatoCreateDTO->getEndUf());
$contatoModel->setInfoEmail($contatoCreateDTO->getInfoEmail());
$contatoModel->setInfoEmailNfe($contatoCreateDTO->getInfoEmailNfe());
$contatoModel->setInfoObservacao($contatoCreateDTO->getInfoObservacao());
$contatoModel->setInfoTelefone($contatoCreateDTO->getInfoTelefone());
$contatoModel->setInfoUsoConsumoIbsCbs($contatoCreateDTO->getInfoUsoConsumoIbsCbs());
$contatoModel->setCreatedAt(new DateTimeImmutable());
return $contatoModel;
} catch(AppException $e) {
throw $e;
}
}
private function validatePFFields(ContatoCreateDTO $contatoCreateDTO, EmpresaModel $empresaModel): void {
$errors = [];
if (($contatoCreateDTO->getDocumentCpf() === '') || !ValidateUtil::cpf(cpf: $contatoCreateDTO->getDocumentCpf())) {
$errors['document_cpf'] = 'CPF informado é inválido.';
throw new AppException(message: 'Dados de contato inválidos.', code: 422, details: $errors);
}
$existingContato = $this->contatoRepo->findOneByConditions(
empresa_id: $empresaModel->getId(),
conditions: [
['field' => 'document_cpf', 'operator' => '=', 'value' => $contatoCreateDTO->getDocumentCpf()]
]
);
if ($existingContato) {
$errors['document_cpf'] = 'CPF já cadastrado para esta empresa.';
throw new AppException(message: 'Dados de contato inválidos.', code: 422, details: $errors);
}
if ($errors) {
throw new AppException(message: 'Dados de contato inválidos.', code: 422, details: $errors);
}
}
private function createPJ(ContatoCreateDTO $contatoCreateDTO, EmpresaModel $empresaModel): ContatoModel {
try {
// Valida os campos específicos para pessoa física
// $this->validatePFFields(contatoCreateDTO: $contatoCreateDTO);
return new ContatoModel();
} catch(AppException $e) {
throw $e;
}
}
}
@@ -0,0 +1,31 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Factories;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Module\v0\Empresa\Repos\EmpresaRepo;
use WorkbloomERP\Module\v0\Contato\Repos\ContatoRepo;
use WorkbloomERP\Module\v0\Empresa\Models\EmpresaModel;
use WorkbloomERP\Module\v0\Contato\DTOs\ContatoCreateDTO;
use WorkbloomERP\Module\v0\Contato\Services\ContatoService;
class ContatoServiceFactory {
public static function makeContatoService(): ContatoService {
$db = new DBService();
return new ContatoService(
db: $db,
contatoRepo: new ContatoRepo(db: $db),
empresaRepo: new EmpresaRepo(db: $db),
);
}
public static function makeContratoFactory(EmpresaModel $empresaModel, ContatoCreateDTO $contatoCreateDTO): ContatoFactory {
$db = new DBService();
return new ContatoFactory(
db: $db,
empresaModel: $empresaModel,
contatoCreateDTO: $contatoCreateDTO,
contatoRepo: new ContatoRepo(db: $db),
empresaRepo: new EmpresaRepo(db: $db),
);
}
}
@@ -0,0 +1,312 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Models;
use DateTimeImmutable;
use WorkbloomERP\Exceptions\AppException;
class ContatoModel {
public function __construct(
private ?int $id = null,
private ?string $uuid = null,
private ?int $empresa_id = null,
private ?bool $is_active = null,
private ?string $tipo = null,
private ?string $nome_empresarial = null,
private ?string $nome_fantasia = null,
private ?string $personalidade = null,
private ?string $document_cpf = null,
private ?string $document_cnpj = null,
private ?int $regime_tributario = null,
private ?int $contribuinte_icms = null,
private ?string $orgao_publico = null,
private ?string $document_ie = null,
private ?string $document_im = null,
private ?string $document_is = null,
private ?string $end_pais = null,
private ?string $end_cep = null,
private ?string $end_ibge = null,
private ?string $end_logradouro = null,
private ?string $end_numero = null,
private ?string $end_complemento = null,
private ?string $end_bairro = null,
private ?string $end_cidade = null,
private ?string $end_uf = null,
private ?string $info_email = null,
private ?string $info_email_nfe = null,
private ?string $info_observacao = null,
private ?string $info_telefone = null,
private ?int $info_uso_consumo_ibs_cbs = null,
private ?DateTimeImmutable $created_at = null,
private ?DateTimeImmutable $updated_at = null,
private ?DateTimeImmutable $deleted_at = null,
) {}
public function toArray(): array {
return [
'id' => $this->getId(),
'uuid' => $this->getUuid(),
'empresa_id' => $this->getEmpresaId(),
'is_active' => $this->getIsActive(),
'tipo' => $this->getTipo(),
'nome_empresarial' => $this->getNomeEmpresarial(),
'nome_fantasia' => $this->getNomeFantasia(),
'personalidade' => $this->getPersonalidade(),
'document_cpf' => $this->getDocumentCpf(),
'document_cnpj' => $this->getDocumentCnpj(),
'regime_tributario' => $this->getRegimeTributario(),
'contribuinte_icms' => $this->getContribuinteIcms(),
'orgao_publico' => $this->getOrgaoPublico(),
'document_ie' => $this->getDocumentIe(),
'document_im' => $this->getDocumentIm(),
'document_is' => $this->getDocumentIs(),
'end_pais' => $this->getEndPais(),
'end_cep' => $this->getEndCep(),
'end_ibge' => $this->getEndIbge(),
'end_logradouro' => $this->getEndLogradouro(),
'end_numero' => $this->getEndNumero(),
'end_complemento' => $this->getEndComplemento(),
'end_bairro' => $this->getEndBairro(),
'end_cidade' => $this->getEndCidade(),
'end_uf' => $this->getEndUf(),
'info_email' => $this->getInfoEmail(),
'info_email_nfe' => $this->getInfoEmailNfe(),
'info_observacao' => $this->getInfoObservacao(),
'info_telefone' => $this->getInfoTelefone(),
'info_uso_consumo_ibs_cbs' => $this->getInfoUsoConsumoIbsCbs(),
'created_at' => $this->getCreatedAt() ? $this->getCreatedAt()->format(format: 'Y-m-d H:i:s') : null,
'updated_at' => $this->getUpdatedAt() ? $this->getUpdatedAt()->format(format: 'Y-m-d H:i:s') : null,
'deleted_at' => $this->getDeletedAt() ? $this->getDeletedAt()->format(format: 'Y-m-d H:i:s') : null,
];
}
public function setId(?int $id): void {
$this->id = $id;
}
public function getId(): ?int {
return $this->id;
}
public function setUuid(?string $uuid): void {
$this->uuid = $uuid;
}
public function getUuid(): ?string {
return $this->uuid;
}
public function setEmpresaId(?int $empresa_id): void {
$this->empresa_id = $empresa_id;
}
public function getEmpresaId(): ?int {
return $this->empresa_id;
}
public function setIsActive(?bool $is_active): void {
$this->is_active = $is_active;
}
public function getIsActive(): ?bool {
return $this->is_active;
}
public function setTipo(?string $tipo): void {
$this->tipo = $tipo;
}
public function getTipo(): ?string {
return $this->tipo;
}
public function setNomeEmpresarial(?string $nome_empresarial): void {
$this->nome_empresarial = $nome_empresarial;
}
public function getNomeEmpresarial(): ?string {
return $this->nome_empresarial;
}
public function setNomeFantasia(?string $nome_fantasia): void {
$this->nome_fantasia = $nome_fantasia;
}
public function getNomeFantasia(): ?string {
return $this->nome_fantasia;
}
public function setPersonalidade(?string $personalidade): void {
$this->personalidade = $personalidade;
}
public function getPersonalidade(): ?string {
return $this->personalidade;
}
public function setDocumentCpf(?string $document_cpf): void {
$this->document_cpf = $document_cpf;
}
public function getDocumentCpf(): ?string {
return $this->document_cpf;
}
public function setDocumentCnpj(?string $document_cnpj): void {
$this->document_cnpj = $document_cnpj;
}
public function getDocumentCnpj(): ?string {
return $this->document_cnpj;
}
public function setRegimeTributario(?int $regime_tributario): void {
$this->regime_tributario = $regime_tributario;
}
public function getRegimeTributario(): ?int {
return $this->regime_tributario;
}
public function setContribuinteIcms(?int $contribuinte_icms): void {
$this->contribuinte_icms = $contribuinte_icms;
}
public function getContribuinteIcms(): ?int {
return $this->contribuinte_icms;
}
public function setOrgaoPublico(?string $orgao_publico): void {
$this->orgao_publico = $orgao_publico;
}
public function getOrgaoPublico(): ?string {
return $this->orgao_publico;
}
public function setDocumentIe(?string $document_ie): void {
$this->document_ie = $document_ie;
}
public function getDocumentIe(): ?string {
return $this->document_ie;
}
public function setDocumentIm(?string $document_im): void {
$this->document_im = $document_im;
}
public function getDocumentIm(): ?string {
return $this->document_im;
}
public function setDocumentIs(?string $document_is): void {
$this->document_is = $document_is;
}
public function getDocumentIs(): ?string {
return $this->document_is;
}
public function setEndPais(?string $end_pais): void {
$this->end_pais = $end_pais;
}
public function getEndPais(): ?string {
return $this->end_pais;
}
public function setEndCep(?string $end_cep): void {
$this->end_cep = $end_cep;
}
public function getEndCep(): ?string {
return $this->end_cep;
}
public function setEndIbge(?string $end_ibge): void {
$this->end_ibge = $end_ibge;
}
public function getEndIbge(): ?string {
return $this->end_ibge;
}
public function setEndLogradouro(?string $end_logradouro): void {
$this->end_logradouro = $end_logradouro;
}
public function getEndLogradouro(): ?string {
return $this->end_logradouro;
}
public function setEndNumero(?string $end_numero): void {
$this->end_numero = $end_numero;
}
public function getEndNumero(): ?string {
return $this->end_numero;
}
public function setEndComplemento(?string $end_complemento): void {
$this->end_complemento = $end_complemento;
}
public function getEndComplemento(): ?string {
return $this->end_complemento;
}
public function setEndBairro(?string $end_bairro): void {
$this->end_bairro = $end_bairro;
}
public function getEndBairro(): ?string {
return $this->end_bairro;
}
public function setEndCidade(?string $end_cidade): void {
$this->end_cidade = $end_cidade;
}
public function getEndCidade(): ?string {
return $this->end_cidade;
}
public function setEndUf(?string $end_uf): void {
$this->end_uf = $end_uf;
}
public function getEndUf(): ?string {
return $this->end_uf;
}
public function setInfoEmail(?string $info_email): void {
$this->info_email = $info_email;
}
public function getInfoEmail(): ?string {
return $this->info_email;
}
public function setInfoEmailNfe(?string $info_email_nfe): void {
$this->info_email_nfe = $info_email_nfe;
}
public function getInfoEmailNfe(): ?string {
return $this->info_email_nfe;
}
public function setInfoObservacao(?string $info_observacao): void {
$this->info_observacao = $info_observacao;
}
public function getInfoObservacao(): ?string {
return $this->info_observacao;
}
public function setInfoTelefone(?string $info_telefone): void {
$this->info_telefone = $info_telefone;
}
public function getInfoTelefone(): ?string {
return $this->info_telefone;
}
public function setInfoUsoConsumoIbsCbs(?int $info_uso_consumo_ibs_cbs): void {
$this->info_uso_consumo_ibs_cbs = $info_uso_consumo_ibs_cbs;
}
public function getInfoUsoConsumoIbsCbs(): ?int {
return $this->info_uso_consumo_ibs_cbs;
}
public function setCreatedAt(?DateTimeImmutable $created_at): void {
$this->created_at = $created_at;
}
public function getCreatedAt(): ?DateTimeImmutable {
return $this->created_at;
}
public function setUpdatedAt(?DateTimeImmutable $updated_at): void {
$this->updated_at = $updated_at;
}
public function getUpdatedAt(): ?DateTimeImmutable {
return $this->updated_at;
}
public function setDeletedAt(?DateTimeImmutable $deleted_at): void {
$this->deleted_at = $deleted_at;
}
public function getDeletedAt(): ?DateTimeImmutable {
return $this->deleted_at;
}
}
+377
View File
@@ -0,0 +1,377 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Repos;
use DateTimeImmutable;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Module\v0\Contato\Models\ContatoModel;
class ContatoRepo {
protected string $contatoTable = 'contato';
public function __construct(
private DBService $db
) {}
public function insert(ContatoModel $contatoModel): ContatoModel {
$query =
"INSERT INTO {$this->contatoTable} (
uuid,
empresa_id,
is_active,
tipo,
nome_empresarial,
nome_fantasia,
personalidade,
document_cpf,
document_cnpj,
regime_tributario,
contribuinte_icms,
orgao_publico,
document_ie,
document_im,
document_is,
end_pais,
end_cep,
end_ibge,
end_logradouro,
end_numero,
end_complemento,
end_bairro,
end_cidade,
end_uf,
info_email,
info_email_nfe,
info_observacao,
info_telefone,
info_uso_consumo_ibs_cbs,
created_at
) VALUES (
:uuid,
:empresa_id,
:is_active,
:tipo,
:nome_empresarial,
:nome_fantasia,
:personalidade,
:document_cpf,
:document_cnpj,
:regime_tributario,
:contribuinte_icms,
:orgao_publico,
:document_ie,
:document_im,
:document_is,
:end_pais,
:end_cep,
:end_ibge,
:end_logradouro,
:end_numero,
:end_complemento,
:end_bairro,
:end_cidade,
:end_uf,
:info_email,
:info_email_nfe,
:info_observacao,
:info_telefone,
:info_uso_consumo_ibs_cbs,
:created_at
)";
$contatoModel->setCreatedAt(new DateTimeImmutable());
$this->db->execute(
sql: $query,
params: [
':uuid' => $contatoModel->getUuid(),
':empresa_id' => $contatoModel->getEmpresaId(),
':is_active' => $contatoModel->getIsActive(),
':tipo' => $contatoModel->getTipo(),
':nome_empresarial' => $contatoModel->getNomeEmpresarial(),
':nome_fantasia' => $contatoModel->getNomeFantasia(),
':personalidade' => $contatoModel->getPersonalidade(),
':document_cpf' => $contatoModel->getDocumentCpf(),
':document_cnpj' => $contatoModel->getDocumentCnpj(),
':regime_tributario' => $contatoModel->getRegimeTributario(),
':contribuinte_icms' => $contatoModel->getContribuinteIcms(),
':orgao_publico' => $contatoModel->getOrgaoPublico(),
':document_ie' => $contatoModel->getDocumentIe(),
':document_im' => $contatoModel->getDocumentIm(),
':document_is' => $contatoModel->getDocumentIs(),
':end_pais' => $contatoModel->getEndPais(),
':end_cep' => $contatoModel->getEndCep(),
':end_ibge' => $contatoModel->getEndIbge(),
':end_logradouro' => $contatoModel->getEndLogradouro(),
':end_numero' => $contatoModel->getEndNumero(),
':end_complemento' => $contatoModel->getEndComplemento(),
':end_bairro' => $contatoModel->getEndBairro(),
':end_cidade' => $contatoModel->getEndCidade(),
':end_uf' => $contatoModel->getEndUf(),
':info_email' => $contatoModel->getInfoEmail(),
':info_email_nfe' => $contatoModel->getInfoEmailNfe(),
':info_observacao' => $contatoModel->getInfoObservacao(),
':info_telefone' => $contatoModel->getInfoTelefone(),
':info_uso_consumo_ibs_cbs' => $contatoModel->getInfoUsoConsumoIbsCbs(),
':created_at' => $contatoModel->getCreatedAt()->format('Y-m-d H:i:s'),
]
);
$contatoModel->setId(id: $this->db->lastInsertId());
return $contatoModel;
}
public function update(ContatoModel $contatoModel): bool {
$query =
"UPDATE {$this->contatoTable} SET
is_active = :is_active,
tipo = :tipo,
nome_empresarial = :nome_empresarial,
nome_fantasia = :nome_fantasia,
personalidade = :personalidade,
document_cpf = :document_cpf,
document_cnpj = :document_cnpj,
regime_tributario = :regime_tributario,
contribuinte_icms = :contribuinte_icms,
orgao_publico = :orgao_publico,
document_ie = :document_ie,
document_im = :document_im,
document_is = :document_is,
end_pais = :end_pais,
end_cep = :end_cep,
end_ibge = :end_ibge,
end_logradouro = :end_logradouro,
end_numero = :end_numero,
end_complemento = :end_complemento,
end_bairro = :end_bairro,
end_cidade = :end_cidade,
end_uf = :end_uf,
info_email = :info_email,
info_email_nfe = :info_email_nfe,
info_observacao = :info_observacao,
info_telefone = :info_telefone,
info_uso_consumo_ibs_cbs = :info_uso_consumo_ibs_cbs,
updated_at = :updated_at
WHERE id = :id OR uuid = :uuid";
return $this->db->execute(
sql: $query,
params: [
':id' => $contatoModel->getId(),
':uuid' => $contatoModel->getUuid(),
':empresa_id' => $contatoModel->getEmpresaId(),
':is_active' => $contatoModel->getIsActive(),
':tipo' => $contatoModel->getTipo(),
':nome_empresarial' => $contatoModel->getNomeEmpresarial(),
':nome_fantasia' => $contatoModel->getNomeFantasia(),
':personalidade' => $contatoModel->getPersonalidade(),
':document_cpf' => $contatoModel->getDocumentCpf(),
':document_cnpj' => $contatoModel->getDocumentCnpj(),
':regime_tributario' => $contatoModel->getRegimeTributario(),
':contribuinte_icms' => $contatoModel->getContribuinteIcms(),
':orgao_publico' => $contatoModel->getOrgaoPublico(),
':document_ie' => $contatoModel->getDocumentIe(),
':document_im' => $contatoModel->getDocumentIm(),
':document_is' => $contatoModel->getDocumentIs(),
':end_pais' => $contatoModel->getEndPais(),
':end_cep' => $contatoModel->getEndCep(),
':end_ibge' => $contatoModel->getEndIbge(),
':end_logradouro' => $contatoModel->getEndLogradouro(),
':end_numero' => $contatoModel->getEndNumero(),
':end_complemento' => $contatoModel->getEndComplemento(),
':end_bairro' => $contatoModel->getEndBairro(),
':end_cidade' => $contatoModel->getEndCidade(),
':end_uf' => $contatoModel->getEndUf(),
':info_email' => $contatoModel->getInfoEmail(),
':info_email_nfe' => $contatoModel->getInfoEmailNfe(),
':info_observacao' => $contatoModel->getInfoObservacao(),
':info_telefone' => $contatoModel->getInfoTelefone(),
':info_uso_consumo_ibs_cbs' => $contatoModel->getInfoUsoConsumoIbsCbs(),
':updated_at' => $contatoModel->getUpdatedAt()->format('Y-m-d H:i:s'),
]
);
}
public function delete(ContatoModel $contatoModel): bool {
$query =
"UPDATE {$this->contatoTable} SET
deleted_at = :deleted_at
WHERE id = :id OR uuid = :uuid";
return $this->db->execute(
sql: $query,
params: [
':id' => $contatoModel->getId(),
':uuid' => $contatoModel->getUuid(),
':deleted_at' => (new DateTimeImmutable())->format('Y-m-d H:i:s'),
]
);
}
public function findByIdentifier(int $empresa_id, string $identifier, mixed $value) {
$query =
"SELECT
id,
uuid,
empresa_id,
is_active,
nome_empresarial,
nome_fantasia,
personalidade,
document_cpf,
document_cnpj,
regime_tributario,
contribuinte_icms,
orgao_publico,
document_ie,
document_im,
document_is,
end_cep,
end_ibge,
end_logradouro,
end_numero,
end_complemento,
end_bairro,
end_cidade,
end_uf,
info_email,
info_email_nfe,
info_observacao,
info_telefone,
info_uso_consumo_ibs_cbs,
created_at,
updated_at,
deleted_at
FROM {$this->contatoTable}
WHERE empresa_id = :empresa_id
AND {$identifier} = :value
AND deleted_at IS NULL
LIMIT 1";
$result = $this->db->fetchOne(
sql: $query,
params: [
':empresa_id' => $empresa_id,
':value' => $value,
]
);
return $result ? $this->mapToModel($result) : null;
}
public function findOneByConditions(int $empresa_id, array $conditions): ?ContatoModel {
if (empty($conditions)) {
throw new \InvalidArgumentException('O array de condições não pode estar vazio.');
}
// Começa com as condições fixas
$whereClauses = [
'empresa_id = :empresa_id',
'deleted_at IS NULL'
];
$params = [
':empresa_id' => $empresa_id
];
// Adiciona as condições dinâmicas
foreach ($conditions as $condition) {
if (!isset($condition['field'], $condition['value'])) {
throw new \InvalidArgumentException(
"Cada condição deve conter 'field' e 'value'."
);
}
$field = $condition['field'];
$operator = $condition['operator'] ?? '=';
$value = $condition['value'];
$whereClauses[] = "{$field} {$operator} :{$field}";
$params[":{$field}"] = $value;
}
$whereSql = implode(' AND ', $whereClauses);
$query =
"SELECT
id,
uuid,
empresa_id,
is_active,
tipo,
nome_empresarial,
nome_fantasia,
personalidade,
document_cpf,
document_cnpj,
regime_tributario,
contribuinte_icms,
orgao_publico,
document_ie,
document_im,
document_is,
end_pais,
end_cep,
end_ibge,
end_logradouro,
end_numero,
end_complemento,
end_bairro,
end_cidade,
end_uf,
info_email,
info_email_nfe,
info_observacao,
info_telefone,
info_uso_consumo_ibs_cbs,
created_at,
updated_at,
deleted_at
FROM {$this->contatoTable}
WHERE {$whereSql}
LIMIT 1";
$result = $this->db->fetchOne(
sql: $query,
params: $params
);
return $result ? $this->mapToModel($result) : null;
}
private function mapToModel(array $data): ContatoModel {
return new ContatoModel(
id: $data['id'],
uuid: $data['uuid'],
empresa_id: $data['empresa_id'],
is_active: $data['is_active'],
tipo: $data['tipo'],
nome_empresarial: $data['nome_empresarial'],
nome_fantasia: $data['nome_fantasia'],
personalidade: $data['personalidade'],
document_cpf: $data['document_cpf'],
document_cnpj: $data['document_cnpj'],
regime_tributario: $data['regime_tributario'],
contribuinte_icms: $data['contribuinte_icms'],
orgao_publico: $data['orgao_publico'],
document_ie: $data['document_ie'],
document_im: $data['document_im'],
document_is: $data['document_is'],
end_pais: $data['end_pais'],
end_cep: $data['end_cep'],
end_ibge: $data['end_ibge'],
end_logradouro: $data['end_logradouro'],
end_numero: $data['end_numero'],
end_complemento: $data['end_complemento'],
end_bairro: $data['end_bairro'],
end_cidade: $data['end_cidade'],
end_uf: $data['end_uf'],
info_email: $data['info_email'],
info_email_nfe: $data['info_email_nfe'],
info_observacao: $data['info_observacao'],
info_telefone: $data['info_telefone'],
info_uso_consumo_ibs_cbs: $data['info_uso_consumo_ibs_cbs'],
created_at: $data['created_at'] ? new DateTimeImmutable($data['created_at']) : null,
updated_at: $data['updated_at'] ? new DateTimeImmutable($data['updated_at']) : null,
deleted_at: $data['deleted_at'] ? new DateTimeImmutable($data['deleted_at']) : null,
);
}
}
+18
View File
@@ -0,0 +1,18 @@
<?php
use KrothiumAPI\Http\Router;
use WorkbloomERP\Module\v0\Auth\Middlewares\AuthMiddleware;
use WorkbloomERP\Module\v0\Contato\Controllers\ContatoController;
Router::group(
prefix: '/contacts',
callback: function() {
// Endpoint para obter as opções de criação de contato (UF, regime tributário, etc.)
Router::get('/create-options', [ContatoController::class, 'createOptions']);
// Rota para cadastrar um novo contato
Router::post('/create', [ContatoController::class, 'create']);
},
middlewares: [
[AuthMiddleware::class, 'handle']
]
);
@@ -0,0 +1,69 @@
<?php
namespace WorkbloomERP\Module\v0\Contato\Services;
use Ramsey\Uuid\Uuid;
use WorkbloomERP\Utils\SanitizeUtil;
use WorkbloomERP\Utils\ValidateUtil;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Exceptions\AppException;
use WorkbloomERP\Module\v0\Auth\Utils\AuthUtil;
use WorkbloomERP\Module\v0\Empresa\Repos\EmpresaRepo;
use WorkbloomERP\Module\v0\Contato\Repos\ContatoRepo;
use WorkbloomERP\Module\v0\Empresa\Models\EmpresaModel;
use WorkbloomERP\Module\v0\Contato\Models\ContatoModel;
use WorkbloomERP\Module\v0\Contato\DTOs\ContatoCreateDTO;
use WorkbloomERP\Module\v0\Contato\Factories\ContatoServiceFactory;
use WorkbloomERP\Constants\SpedCRTConst;
use WorkbloomERP\Constants\BrasilUfsConst;
use WorkbloomERP\Constants\SpedPaisesConst;
use WorkbloomERP\Module\v0\Contato\Constants\OrgaoPublicoConst;
use WorkbloomERP\Module\v0\Contato\Constants\PersonalidadeConst;
use WorkbloomERP\Module\v0\Contato\Constants\ContribuinteICMSConst;
use WorkbloomERP\Module\v0\Contato\Constants\NFSeConsumoIbsCbsConst;
class ContatoService {
public function __construct (
protected DBService $db,
protected ContatoRepo $contatoRepo,
protected EmpresaRepo $empresaRepo,
) {}
public function createOptions(): array {
return [
'tipo' => PersonalidadeConst::getAll(),
'personalidade' => PersonalidadeConst::getAll(),
'uf' => BrasilUfsConst::getAll(),
'regime_tributario' => SpedCRTConst::getAll(),
'contribuinte_icms' => ContribuinteICMSConst::getAll(),
'info_uso_consumo_ibs_cbs' => NFSeConsumoIbsCbsConst::getAll(),
'orgao_publico' => OrgaoPublicoConst::getAll(),
'paises_sped' => SpedPaisesConst::getAll()
];
}
public function create(ContatoCreateDTO $contatoCreateDTO): array {
try {
return $this->db->transaction(
callback: function() use ($contatoCreateDTO) {
// Pega as informações da sessão para associar o contato à empresa correta
$empresaData = AuthUtil::readSession(key: 'empresa_data');
// Verifica se a empresa existe
$empresaModel = $this->empresaRepo->findByIdentifier(identifier: 'uuid', value: $empresaData['uuid']);
// Cria a fábrica de contato e valida os dados de criação do contato
$contatoFactory = ContatoServiceFactory::makeContratoFactory(empresaModel: $empresaModel, contatoCreateDTO: $contatoCreateDTO);
$contatoModel = $contatoFactory->create(contatoCreateDTO: $contatoCreateDTO, empresaModel: $empresaModel);
return [
'response_code' => 201,
'message' => 'Contato criado com sucesso.',
'output' => ['data' => $contatoModel->toArray()]
];
}
);
} catch(AppException $e) {
throw $e;
}
}
}
@@ -0,0 +1,212 @@
<?php
namespace WorkbloomERP\Module\v0\Empresa\Models;
use DateTimeImmutable;
class EmpresaModel {
public function __construct(
private ?int $id = null,
private ?string $uuid = null,
private ?bool $is_active = null,
private ?string $nome_empresarial = null,
private ?string $nome_fantasia = null,
private ?string $tipo = null,
private ?int $matriz_id = null,
private ?string $document_cnpj = null,
private ?string $document_ie = null,
private ?string $document_im = null,
private ?string $regime_tributario = null,
private ?string $end_cep = null,
private ?int $end_ibge = null,
private ?string $end_logradouro = null,
private ?string $end_numero = null,
private ?string $end_complemento = null,
private ?string $end_bairro = null,
private ?string $end_cidade = null,
private ?string $end_uf = null,
private ?DateTimeImmutable $created_at = null,
private ?DateTimeImmutable $updated_at = null,
private ?DateTimeImmutable $deleted_at = null,
) {}
public function toArray(): array {
return [
'id' => $this->getId(),
'uuid' => $this->getUuid(),
'is_active' => $this->getIsActive(),
'nome_empresarial' => $this->getNomeEmpresarial(),
'nome_fantasia' => $this->getNomeFantasia(),
'tipo' => $this->getTipo(),
'matriz_id' => $this->getMatrizId(),
'document_cnpj' => $this->getDocumentCnpj(),
'document_ie' => $this->getDocumentIe(),
'document_im' => $this->getDocumentIm(),
'regime_tributario' => $this->getRegimeTributario(),
'end_cep' => $this->getEndCep(),
'end_ibge' => $this->getEndIbge(),
'end_logradouro' => $this->getEndLogradouro(),
'end_numero' => $this->getEndNumero(),
'end_complemento' => $this->getEndComplemento(),
'end_bairro' => $this->getEndBairro(),
'end_cidade' => $this->getEndCidade(),
'end_uf' => $this->getEndUf(),
'created_at' => $this->getCreatedAt() ? $this->getCreatedAt()->format('Y-m-d H:i:s') : null,
'updated_at' => $this->getUpdatedAt() ? $this->getUpdatedAt()->format('Y-m-d H:i:s') : null,
'deleted_at' => $this->getDeletedAt() ? $this->getDeletedAt()->format('Y-m-d H:i:s') : null,
];
}
public function setId(?int $id): void {
$this->id = $id;
}
public function getId(): ?int {
return $this->id;
}
public function setUuid(?string $uuid): void {
$this->uuid = $uuid;
}
public function getUuid(): ?string {
return $this->uuid;
}
public function setIsActive(?bool $is_active): void {
$this->is_active = $is_active;
}
public function getIsActive(): ?bool {
return $this->is_active;
}
public function setNomeEmpresarial(?string $nome_empresarial): void {
$this->nome_empresarial = $nome_empresarial;
}
public function getNomeEmpresarial(): ?string {
return $this->nome_empresarial;
}
public function setNomeFantasia(?string $nome_fantasia): void {
$this->nome_fantasia = $nome_fantasia;
}
public function getNomeFantasia(): ?string {
return $this->nome_fantasia;
}
public function setTipo(?string $tipo): void {
$this->tipo = $tipo;
}
public function getTipo(): ?string {
return $this->tipo;
}
public function setMatrizId(?int $matriz_id): void {
$this->matriz_id = $matriz_id;
}
public function getMatrizId(): ?int {
return $this->matriz_id;
}
public function setDocumentCnpj(?string $document_cnpj): void {
$this->document_cnpj = $document_cnpj;
}
public function getDocumentCnpj(): ?string {
return $this->document_cnpj;
}
public function setDocumentIe(?string $document_ie): void {
$this->document_ie = $document_ie;
}
public function getDocumentIe(): ?string {
return $this->document_ie;
}
public function setDocumentIm(?string $document_im): void {
$this->document_im = $document_im;
}
public function getDocumentIm(): ?string {
return $this->document_im;
}
public function setRegimeTributario(?string $regime_tributario): void {
$this->regime_tributario = $regime_tributario;
}
public function getRegimeTributario(): ?string {
return $this->regime_tributario;
}
public function setEndCep(?string $end_cep): void {
$this->end_cep = $end_cep;
}
public function getEndCep(): ?string {
return $this->end_cep;
}
public function setEndIbge(?int $end_ibge): void {
$this->end_ibge = $end_ibge;
}
public function getEndIbge(): ?int {
return $this->end_ibge;
}
public function setEndLogradouro(?string $end_logradouro): void {
$this->end_logradouro = $end_logradouro;
}
public function getEndLogradouro(): ?string {
return $this->end_logradouro;
}
public function setEndNumero(?string $end_numero): void {
$this->end_numero = $end_numero;
}
public function getEndNumero(): ?string {
return $this->end_numero;
}
public function setEndComplemento(?string $end_complemento): void {
$this->end_complemento = $end_complemento;
}
public function getEndComplemento(): ?string {
return $this->end_complemento;
}
public function setEndBairro(?string $end_bairro): void {
$this->end_bairro = $end_bairro;
}
public function getEndBairro(): ?string {
return $this->end_bairro;
}
public function setEndCidade(?string $end_cidade): void {
$this->end_cidade = $end_cidade;
}
public function getEndCidade(): ?string {
return $this->end_cidade;
}
public function setEndUf(?string $end_uf): void {
$this->end_uf = $end_uf;
}
public function getEndUf(): ?string {
return $this->end_uf;
}
public function setCreatedAt(?DateTimeImmutable $created_at): void {
$this->created_at = $created_at;
}
public function getCreatedAt(): ?DateTimeImmutable {
return $this->created_at;
}
public function setUpdatedAt(?DateTimeImmutable $updated_at): void {
$this->updated_at = $updated_at;
}
public function getUpdatedAt(): ?DateTimeImmutable {
return $this->updated_at;
}
public function setDeletedAt(?DateTimeImmutable $deleted_at): void {
$this->deleted_at = $deleted_at;
}
public function getDeletedAt(): ?DateTimeImmutable {
return $this->deleted_at;
}
}
+268
View File
@@ -0,0 +1,268 @@
<?php
namespace WorkbloomERP\Module\v0\Empresa\Repos;
use DateTimeImmutable;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Module\v0\Empresa\Models\EmpresaModel;
class EmpresaRepo {
protected string $empresaTable = 'empresa';
public function __construct(
protected DBService $db
) {}
public function insert(EmpresaModel $empresaModel): ?EmpresaModel {
$query =
"INSERT INTO {$this->empresaTable} (
uuid,
is_active,
nome_empresarial,
nome_fantasia,
tipo,
matriz_id,
document_cnpj,
document_ie,
document_im,
regime_tributario,
end_cep,
end_ibge,
end_logradouro,
end_numero,
end_complemento,
end_bairro,
end_cidade,
end_uf,
created_at
) VALUES (
:uuid,
:is_active,
:nome_empresarial,
:nome_fantasia,
:tipo,
:matriz_id,
:document_cnpj,
:document_ie,
:document_im,
:regime_tributario,
:end_cep,
:end_ibge,
:end_logradouro,
:end_numero,
:end_complemento,
:end_bairro,
:end_cidade,
:end_uf,
:created_at
)";
$empresaModel->setCreatedAt(new DateTimeImmutable());
$this->db->execute(
sql: $query,
params: [
':uuid' => $empresaModel->getUuid(),
':is_active' => $empresaModel->getIsActive(),
':nome_empresarial' => $empresaModel->getNomeEmpresarial(),
':nome_fantasia' => $empresaModel->getNomeFantasia(),
':tipo' => $empresaModel->getTipo(),
':matriz_id' => $empresaModel->getMatrizId(),
':document_cnpj' => $empresaModel->getDocumentCnpj(),
':document_ie' => $empresaModel->getDocumentIe(),
':document_im' => $empresaModel->getDocumentIm(),
':regime_tributario' => $empresaModel->getRegimeTributario(),
':end_cep' => $empresaModel->getEndCep(),
':end_ibge' => $empresaModel->getEndIbge(),
':end_logradouro' => $empresaModel->getEndLogradouro(),
':end_numero' => $empresaModel->getEndNumero(),
':end_complemento' => $empresaModel->getEndComplemento(),
':end_bairro' => $empresaModel->getEndBairro(),
':end_cidade' => $empresaModel->getEndCidade(),
':end_uf' => $empresaModel->getEndUf(),
':created_at' => $empresaModel->getCreatedAt()->format('Y-m-d H:i:s'),
]
);
$empresaModel->setId($this->db->lastInsertId());
return $empresaModel;
}
public function update(EmpresaModel $empresaModel): bool {
$query =
"UPDATE {$this->empresaTable} SET
is_active = :is_active,
nome_empresarial = :nome_empresarial,
nome_fantasia = :nome_fantasia,
tipo = :tipo,
matriz_id = :matriz_id,
document_cnpj = :document_cnpj,
document_ie = :document_ie,
document_im = :document_im,
regime_tributario = :regime_tributario,
end_cep = :end_cep,
end_ibge = :end_ibge,
end_logradouro = :end_logradouro,
end_numero = :end_numero,
end_complemento = :end_complemento,
end_bairro = :end_bairro,
end_cidade = :end_cidade,
end_uf = :end_uf,
updated_at = :updated_at
WHERE id = :id OR uuid = :uuid";
$empresaModel->setUpdatedAt(new DateTimeImmutable());
return $this->db->execute(
sql: $query,
params: [
':id' => $empresaModel->getId(),
':uuid' => $empresaModel->getUuid(),
':is_active' => $empresaModel->getIsActive(),
':nome_empresarial' => $empresaModel->getNomeEmpresarial(),
':nome_fantasia' => $empresaModel->getNomeFantasia(),
':tipo' => $empresaModel->getTipo(),
':matriz_id' => $empresaModel->getMatrizId(),
':document_cnpj' => $empresaModel->getDocumentCnpj(),
':document_ie' => $empresaModel->getDocumentIe(),
':document_im' => $empresaModel->getDocumentIm(),
':regime_tributario' => $empresaModel->getRegimeTributario(),
':end_cep' => $empresaModel->getEndCep(),
':end_ibge' => $empresaModel->getEndIbge(),
':end_logradouro' => $empresaModel->getEndLogradouro(),
':end_numero' => $empresaModel->getEndNumero(),
':end_complemento' => $empresaModel->getEndComplemento(),
':end_bairro' => $empresaModel->getEndBairro(),
':end_cidade' => $empresaModel->getEndCidade(),
':end_uf' => $empresaModel->getEndUf(),
':updated_at' => $empresaModel->getUpdatedAt()->format('Y-m-d H:i:s'),
]
);
}
public function delete(EmpresaModel $empresaModel): bool {
$query =
"UPDATE {$this->empresaTable} SET
deleted_at = :deleted_at
WHERE id = :id OR uuid = :uuid";
$empresaModel->setDeletedAt(new DateTimeImmutable());
return $this->db->execute(
sql: $query,
params: [
':id' => $empresaModel->getId(),
':uuid' => $empresaModel->getUuid(),
':deleted_at' => $empresaModel->getDeletedAt()->format('Y-m-d H:i:s'),
]
);
}
public function findByIdentifier(string $identifier, mixed $value): ?EmpresaModel {
$query =
"SELECT
id,
uuid,
is_active,
nome_empresarial,
nome_fantasia,
tipo,
matriz_id,
document_cnpj,
document_ie,
document_im,
regime_tributario,
end_cep,
end_ibge,
end_logradouro,
end_numero,
end_complemento,
end_bairro,
end_cidade,
end_uf,
created_at,
updated_at,
deleted_at
FROM {$this->empresaTable}
WHERE $identifier = :value
AND deleted_at IS NULL
LIMIT 1";
$result = $this->db->fetchOne(
sql: $query,
params: [
':value' => $value
]
);
return $result ? $this->mapToModel($result) : null;
}
public function findAllByMatrizId(int $matriz_id): array {
$query =
"SELECT
id,
uuid,
is_active,
nome_empresarial,
nome_fantasia,
tipo,
matriz_id,
document_cnpj,
document_ie,
document_im,
regime_tributario,
end_cep,
end_ibge,
end_logradouro,
end_numero,
end_complemento,
end_bairro,
end_cidade,
end_uf,
created_at,
updated_at,
deleted_at
FROM {$this->empresaTable}
WHERE (
id = :matriz_id OR
matriz_id = :matriz_id
)
AND deleted_at IS NULL";
$results = $this->db->fetchAll(
sql: $query,
params: [
':matriz_id' => $matriz_id
]
);
return array_map(fn($data) => $this->mapToModel($data), $results);
}
private function mapToModel(array $data): EmpresaModel {
return new EmpresaModel(
id: $data['id'],
uuid: $data['uuid'],
is_active: $data['is_active'],
nome_empresarial: $data['nome_empresarial'],
nome_fantasia: $data['nome_fantasia'],
tipo: $data['tipo'],
matriz_id: $data['matriz_id'],
document_cnpj: $data['document_cnpj'],
document_ie: $data['document_ie'],
document_im: $data['document_im'],
regime_tributario: $data['regime_tributario'],
end_cep: $data['end_cep'],
end_ibge: $data['end_ibge'],
end_logradouro: $data['end_logradouro'],
end_numero: $data['end_numero'],
end_complemento: $data['end_complemento'],
end_bairro: $data['end_bairro'],
end_cidade: $data['end_cidade'],
end_uf: $data['end_uf'],
created_at: $data['created_at'] ? new DateTimeImmutable($data['created_at']) : null,
updated_at: $data['updated_at'] ? new DateTimeImmutable($data['updated_at']) : null,
deleted_at: $data['deleted_at'] ? new DateTimeImmutable($data['deleted_at']) : null,
);
}
}
@@ -0,0 +1,32 @@
<?php
namespace WorkbloomERP\Module\v0\Usuario\Models;
use DateTimeImmutable;
class UsuarioEmpresaModel {
public function __construct(
private ?int $usuario_id = null,
private ?int $empresa_id = null,
) {}
public function toArray(): array {
return [
'usuario_id' => $this->getUsuarioId(),
'empresa_id' => $this->getEmpresaId(),
];
}
public function getUsuarioId(): ?int {
return $this->usuario_id;
}
public function setUsuarioId(?int $usuario_id): void {
$this->usuario_id = $usuario_id;
}
public function getEmpresaId(): ?int {
return $this->empresa_id;
}
public function setEmpresaId(?int $empresa_id): void {
$this->empresa_id = $empresa_id;
}
}
@@ -0,0 +1,113 @@
<?php
namespace WorkbloomERP\Module\v0\Usuario\Models;
use DateTimeImmutable;
class UsuarioModel {
public function __construct(
private ?int $id = null,
private ?string $uuid = null,
private ?bool $is_active = null,
private ?bool $is_root = null,
private ?string $nome_completo = null,
private ?string $nome_usuario = null,
private ?string $email = null,
private ?string $senha_hash = null,
private ?DateTimeImmutable $created_at = null,
private ?DateTimeImmutable $updated_at = null,
private ?DateTimeImmutable $deleted_at = null,
) {}
public function toArray(): array {
return [
'id' => $this->getId(),
'uuid' => $this->getUuid(),
'is_active' => $this->getIsActive(),
'is_root' => $this->getIsRoot(),
'nome_completo' => $this->getNomeCompleto(),
'nome_usuario' => $this->getNomeUsuario(),
'email' => $this->getEmail(),
'senha_hash' => $this->getSenhaHash(),
'created_at' => $this->getCreatedAt() ? $this->getCreatedAt()->format('Y-m-d H:i:s') : null,
'updated_at' => $this->getUpdatedAt() ? $this->getUpdatedAt()->format('Y-m-d H:i:s') : null,
'deleted_at' => $this->getDeletedAt() ? $this->getDeletedAt()->format('Y-m-d H:i:s') : null,
];
}
public function setId(?int $id): void {
$this->id = $id;
}
public function getId(): ?int {
return $this->id;
}
public function setUuid(?string $uuid): void {
$this->uuid = $uuid;
}
public function getUuid(): ?string {
return $this->uuid;
}
public function setIsActive(?bool $is_active): void {
$this->is_active = $is_active;
}
public function getIsActive(): ?bool {
return $this->is_active;
}
public function setIsRoot(?bool $is_root): void {
$this->is_root = $is_root;
}
public function getIsRoot(): ?bool {
return $this->is_root;
}
public function setNomeCompleto(?string $nome_completo): void {
$this->nome_completo = $nome_completo;
}
public function getNomeCompleto(): ?string {
return $this->nome_completo;
}
public function setNomeUsuario(?string $nome_usuario): void {
$this->nome_usuario = $nome_usuario;
}
public function getNomeUsuario(): ?string {
return $this->nome_usuario;
}
public function setEmail(?string $email): void {
$this->email = $email;
}
public function getEmail(): ?string {
return $this->email;
}
public function setSenhaHash(?string $senha_hash): void {
$this->senha_hash = $senha_hash;
}
public function getSenhaHash(): ?string {
return $this->senha_hash;
}
public function setCreatedAt(?DateTimeImmutable $created_at): void {
$this->created_at = $created_at;
}
public function getCreatedAt(): ?DateTimeImmutable {
return $this->created_at;
}
public function setUpdatedAt(?DateTimeImmutable $updated_at): void {
$this->updated_at = $updated_at;
}
public function getUpdatedAt(): ?DateTimeImmutable {
return $this->updated_at;
}
public function setDeletedAt(?DateTimeImmutable $deleted_at): void {
$this->deleted_at = $deleted_at;
}
public function getDeletedAt(): ?DateTimeImmutable {
return $this->deleted_at;
}
}
@@ -0,0 +1,86 @@
<?php
namespace WorkbloomERP\Module\v0\Usuario\Models;
use DateTimeImmutable;
class UsuarioSessionModel {
public function __construct(
private ?int $id = null,
private ?string $uuid = null,
private ?int $usuario_id = null,
private ?string $user_agent = null,
private ?string $ip_address = null,
private ?string $token_hash = null,
private ?DateTimeImmutable $created_at = null,
private ?DateTimeImmutable $revoked_at = null,
) {}
public function toArray(): array {
return [
'id' => $this->getId(),
'uuid' => $this->getUuid(),
'usuario_id' => $this->getUsuarioId(),
'user_agent' => $this->getUserAgent(),
'ip_address' => $this->getIpAddress(),
'token_hash' => $this->getTokenHash(),
'created_at' => $this->getCreatedAt() ? $this->getCreatedAt()->format('Y-m-d H:i:s') : null,
'revoked_at' => $this->getRevokedAt() ? $this->getRevokedAt()->format('Y-m-d H:i:s') : null,
];
}
public function setId(?int $id): void {
$this->id = $id;
}
public function getId(): ?int {
return $this->id;
}
public function setUuid(?string $uuid): void {
$this->uuid = $uuid;
}
public function getUuid(): ?string {
return $this->uuid;
}
public function setUsuarioId(?int $usuario_id): void {
$this->usuario_id = $usuario_id;
}
public function getUsuarioId(): ?int {
return $this->usuario_id;
}
public function setUserAgent(?string $user_agent): void {
$this->user_agent = $user_agent;
}
public function getUserAgent(): ?string {
return $this->user_agent;
}
public function setIpAddress(?string $ip_address): void {
$this->ip_address = $ip_address;
}
public function getIpAddress(): ?string {
return $this->ip_address;
}
public function setTokenHash(?string $token_hash): void {
$this->token_hash = $token_hash;
}
public function getTokenHash(): ?string {
return $this->token_hash;
}
public function setCreatedAt(?DateTimeImmutable $created_at): void {
$this->created_at = $created_at;
}
public function getCreatedAt(): ?DateTimeImmutable {
return $this->created_at;
}
public function setRevokedAt(?DateTimeImmutable $revoked_at): void {
$this->revoked_at = $revoked_at;
}
public function getRevokedAt(): ?DateTimeImmutable {
return $this->revoked_at;
}
}
@@ -0,0 +1,77 @@
<?php
namespace WorkbloomERP\Module\v0\Usuario\Repos;
use DateTimeImmutable;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioEmpresaModel;
class UsuarioEmpresaRepo {
protected string $usuarioEmpresaTable = 'shared.usuario_empresa';
public function __construct(
private DBService $db
) {}
public function insert(UsuarioEmpresaModel $usuarioEmpresaModel): bool {
$query =
"INSERT INTO {$this->usuarioEmpresaTable} (
usuario_id,
empresa_id
) VALUES (
:usuario_id,
:empresa_id
)";
return $this->db->execute(
sql: $query,
params: [
'usuario_id' => $usuarioEmpresaModel->getUsuarioId(),
'empresa_id' => $usuarioEmpresaModel->getEmpresaId()
]
);
}
public function delete(UsuarioEmpresaModel $usuarioEmpresaModel): bool {
$query =
"DELETE FROM {$this->usuarioEmpresaTable} WHERE usuario_id = :usuario_id AND empresa_id = :empresa_id";
return $this->db->execute(
sql: $query,
params: [
'usuario_id' => $usuarioEmpresaModel->getUsuarioId(),
'empresa_id' => $usuarioEmpresaModel->getEmpresaId()
]
);
}
public function findAllByUsuarioId(int $usuario_id): array {
$query =
"SELECT
usuario_id,
empresa_id
FROM {$this->usuarioEmpresaTable}
WHERE usuario_id = :usuario_id";
return $this->db->fetchAll(
sql: $query,
params: [
'usuario_id' => $usuario_id
]
);
}
public function checkAssociationByUsuarioIdAndEmpresaId(int $usuario_id, int $empresa_id): bool {
$query =
"SELECT 1 FROM {$this->usuarioEmpresaTable} WHERE usuario_id = :usuario_id AND empresa_id = :empresa_id";
$result = $this->db->fetchOne(
sql: $query,
params: [
'usuario_id' => $usuario_id,
'empresa_id' => $empresa_id
]
);
return !empty($result);
}
}
+182
View File
@@ -0,0 +1,182 @@
<?php
namespace WorkbloomERP\Module\v0\Usuario\Repos;
use DateTimeImmutable;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioModel;
class UsuarioRepo {
protected string $usuarioTable = 'usuario';
public function __construct(
private DBService $db
) {}
public function insert(UsuarioModel $usuarioModel): ?UsuarioModel {
$query =
"INSERT INTO {$this->usuarioTable} (
uuid,
is_active,
is_root,
nome_completo,
nome_usuario,
email,
senha_hash,
created_at
) VALUES (
:uuid,
:is_active,
:is_root,
:nome_completo,
:nome_usuario,
:email,
:senha_hash,
:created_at
)";
$usuarioModel->setCreatedAt(created_at: new DateTimeImmutable());
$this->db->execute(
sql: $query,
params: [
'uuid' => $usuarioModel->getUuid(),
'is_active' => $usuarioModel->getIsActive(),
'is_root' => $usuarioModel->getIsRoot(),
'nome_completo' => $usuarioModel->getNomeCompleto(),
'nome_usuario' => $usuarioModel->getNomeUsuario(),
'email' => $usuarioModel->getEmail(),
'senha_hash' => $usuarioModel->getSenhaHash(),
'created_at' => $usuarioModel->getCreatedAt()->format(format: 'Y-m-d H:i:s')
]
);
$usuarioModel->setId(id: $this->db->lastInsertId());
return $usuarioModel;
}
public function update(UsuarioModel $usuarioModel): ?UsuarioModel {
$query =
"UPDATE {$this->usuarioTable} SET
is_active = :is_active,
is_root = :is_root,
nome_completo = :nome_completo,
nome_usuario = :nome_usuario,
email = :email,
senha_hash = :senha_hash,
updated_at = :updated_at
WHERE id = :id OR uuid = :uuid";
$usuarioModel->setUpdatedAt(updated_at: new DateTimeImmutable());
$this->db->execute(
sql: $query,
params: [
':id' => $usuarioModel->getId(),
':uuid' => $usuarioModel->getUuid(),
':is_active' => $usuarioModel->getIsActive(),
':is_root' => $usuarioModel->getIsRoot(),
':nome_completo' => $usuarioModel->getNomeCompleto(),
':nome_usuario' => $usuarioModel->getNomeUsuario(),
':email' => $usuarioModel->getEmail(),
':senha_hash' => $usuarioModel->getSenhaHash(),
':updated_at' => $usuarioModel->getUpdatedAt()->format(format: 'Y-m-d H:i:s')
]
);
return $usuarioModel;
}
public function delete(UsuarioModel $usuarioModel): ?UsuarioModel {
$query =
"UPDATE {$this->usuarioTable} SET
is_active = 0,
deleted_at = :deleted_at
WHERE id = :id OR uuid = :uuid";
$usuarioModel->setDeletedAt(deleted_at: new DateTimeImmutable());
$this->db->execute(
sql: $query,
params: [
':id' => $usuarioModel->getId(),
':uuid' => $usuarioModel->getUuid(),
':deleted_at' => $usuarioModel->getDeletedAt()->format(format: 'Y-m-d H:i:s')
]
);
return $usuarioModel;
}
public function findByIdentifier(string $identifier, mixed $value): ?UsuarioModel {
$query =
"SELECT
id,
uuid,
is_active,
is_root,
nome_completo,
nome_usuario,
email,
senha_hash,
created_at,
updated_at,
deleted_at
FROM {$this->usuarioTable}
WHERE {$identifier} = :value
AND deleted_at IS NULL
LIMIT 1";
$result = $this->db->fetchOne(
sql: $query,
params: [
':value' => $value
]
);
return $result ? $this->mapToModel($result) : null;
}
public function findByLogin(string $login): ?UsuarioModel {
$query =
"SELECT
id,
uuid,
is_active,
is_root,
nome_completo,
nome_usuario,
email,
senha_hash,
created_at,
updated_at,
deleted_at
FROM {$this->usuarioTable}
WHERE (nome_usuario = :login OR email = :login)
AND deleted_at IS NULL
LIMIT 1";
$result = $this->db->fetchOne(
sql: $query,
params: [
':login' => $login
]
);
return $result ? $this->mapToModel($result) : null;
}
private function mapToModel(array $data): UsuarioModel {
return new UsuarioModel(
id: $data['id'],
uuid: $data['uuid'],
is_active: $data['is_active'],
is_root: $data['is_root'],
nome_completo: $data['nome_completo'],
nome_usuario: $data['nome_usuario'],
email: $data['email'],
senha_hash: $data['senha_hash'],
created_at: $data['created_at'] ? new DateTimeImmutable($data['created_at']) : null,
updated_at: $data['updated_at'] ? new DateTimeImmutable($data['updated_at']) : null,
deleted_at: $data['deleted_at'] ? new DateTimeImmutable($data['deleted_at']) : null,
);
}
}
@@ -0,0 +1,110 @@
<?php
namespace WorkbloomERP\Module\v0\Usuario\Repos;
use DateTimeImmutable;
use WorkbloomERP\Services\DBService;
use WorkbloomERP\Module\v0\Usuario\Models\UsuarioSessionModel;
class UsuarioSessionRepo {
protected string $usuarioSessionTable = 'usuario_session';
public function __construct(
private DBService $db
) {}
public function insert(UsuarioSessionModel $usuarioSessionModel): ?UsuarioSessionModel {
$query =
"INSERT INTO {$this->usuarioSessionTable} (
uuid,
usuario_id,
user_agent,
ip_address,
token_hash,
created_at
) VALUES (
:uuid,
:usuario_id,
:user_agent,
:ip_address,
:token_hash,
:created_at
)";
$usuarioSessionModel->setCreatedAt(new DateTimeImmutable());
$this->db->execute(
sql: $query,
params: [
':uuid' => $usuarioSessionModel->getUuid(),
':usuario_id' => $usuarioSessionModel->getUsuarioId(),
':user_agent' => $usuarioSessionModel->getUserAgent(),
':ip_address' => $usuarioSessionModel->getIpAddress(),
':token_hash' => $usuarioSessionModel->getTokenHash(),
':created_at' => $usuarioSessionModel->getCreatedAt()->format('Y-m-d H:i:s')
]
);
$usuarioSessionModel->setId($this->db->lastInsertId());
return $usuarioSessionModel;
}
public function update(UsuarioSessionModel $usuarioSessionModel): bool {
$query =
"UPDATE {$this->usuarioSessionTable} SET
usuario_id = :usuario_id,
user_agent = :user_agent,
ip_address = :ip_address,
token_hash = :token_hash,
revoked_at = :revoked_at
WHERE id = :id OR uuid = :uuid";
return $this->db->execute(
sql: $query,
params: [
':id' => $usuarioSessionModel->getId(),
':uuid' => $usuarioSessionModel->getUuid(),
':usuario_id' => $usuarioSessionModel->getUsuarioId(),
':user_agent' => $usuarioSessionModel->getUserAgent(),
':ip_address' => $usuarioSessionModel->getIpAddress(),
':token_hash' => $usuarioSessionModel->getTokenHash(),
':revoked_at' => $usuarioSessionModel->getRevokedAt() ? $usuarioSessionModel->getRevokedAt()->format('Y-m-d H:i:s') : null
]
);
}
public function findByIdentifier(string $identifier, mixed $value): ?UsuarioSessionModel {
$query =
"SELECT
id,
uuid,
usuario_id,
user_agent,
ip_address,
token_hash,
created_at,
revoked_at
FROM {$this->usuarioSessionTable}
WHERE $identifier = :value
LIMIT 1";
$result = $this->db->fetchOne(
sql: $query,
params: [':value' => $value]
);
return $result ? $this->mapToModel($result) : null;
}
private function mapToModel(array $data) {
return new UsuarioSessionModel(
id: $data['id'],
uuid: $data['uuid'],
usuario_id: $data['usuario_id'],
user_agent: $data['user_agent'],
ip_address: $data['ip_address'],
token_hash: $data['token_hash'],
created_at: $data['created_at'] ? new DateTimeImmutable($data['created_at']) : null,
revoked_at: $data['revoked_at'] ? new DateTimeImmutable($data['revoked_at']) : null,
);
}
}