first commit
This commit is contained in:
@@ -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);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -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']]);
|
||||
}
|
||||
);
|
||||
@@ -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;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -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;
|
||||
}
|
||||
}
|
||||
@@ -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,
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -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;
|
||||
}
|
||||
}
|
||||
@@ -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);
|
||||
}
|
||||
}
|
||||
@@ -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,
|
||||
);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user