CompanyWithUserController.php 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. <?php
  2. namespace Controllers;
  3. use Libs\BashExecutor;
  4. use Libs\ResponseLib;
  5. use Models\CompanyModel;
  6. use Models\UserModel;
  7. use Services\TshieldService;
  8. use Psr\Http\Message\ServerRequestInterface;
  9. use Respect\Validation\Validator as val;
  10. use Respect\Validation\Exceptions\ValidationException;
  11. use React\Http\Message\Response;
  12. class CompanyWithUserController
  13. {
  14. private CompanyModel $companyModel;
  15. private UserModel $userModel;
  16. private TshieldService $tshieldService;
  17. public function __construct()
  18. {
  19. $this->companyModel = new CompanyModel();
  20. $this->userModel = new UserModel();
  21. $this->tshieldService = new TshieldService();
  22. }
  23. public function __invoke(ServerRequestInterface $request)
  24. {
  25. $body = json_decode((string)$request->getBody(), true) ?? [];
  26. try {
  27. val::key('company_name', val::stringType()->notEmpty()->length(1, 255))
  28. ->key('company_user', val::stringType()->notEmpty()->length(1, 255))
  29. ->key('username', val::stringType()->notEmpty()->length(1, 100))
  30. ->key('email', val::email())
  31. ->key('password', val::stringType()->length(8, null))
  32. ->key('phone', val::stringType()->notEmpty()->length(1, 50))
  33. ->key('address', val::stringType()->notEmpty()->length(1, 255))
  34. ->key('city', val::stringType()->notEmpty()->length(1, 100))
  35. ->key('state', val::stringType()->notEmpty()->length(1, 100))
  36. ->key('zip', val::stringType()->notEmpty()->length(1, 20))
  37. ->key('country', val::stringType()->notEmpty()->length(1, 100))
  38. ->key('kyc', val::intType(), false)
  39. ->key('birthdate', val::intType())
  40. ->key('cpf', val::stringType()->notEmpty()->length(1, 11))
  41. ->key('cnpj', val::stringType()->notEmpty()->length(1, 14))
  42. ->assert($body);
  43. } catch (ValidationException $e) {
  44. return ResponseLib::sendFail("Validation failed: " . $e->getFullMessage(), [], "E_VALIDATE")->withStatus(400);
  45. }
  46. try {
  47. $pdo = $GLOBALS['pdo'];
  48. $pdo->beginTransaction();
  49. $companyId = $this->companyModel->createCompany($body['company_name'], $body['cnpj'], 'a');
  50. $roleId = 1;
  51. $chk = $pdo->prepare('SELECT 1 FROM "role" WHERE role_id = :rid');
  52. $chk->execute(['rid' => $roleId]);
  53. if (!$chk->fetchColumn()) {
  54. $pdo->rollBack();
  55. return ResponseLib::sendFail('Default role_id 1 not found', [], 'E_DATABASE')->withStatus(500);
  56. }
  57. $userPayload = [
  58. 'username' => $body['username'],
  59. 'email' => $body['email'],
  60. 'password' => $body['password'],
  61. 'phone' => $body['phone'],
  62. 'address' => $body['address'],
  63. 'city' => $body['city'],
  64. 'state' => $body['state'],
  65. 'zip' => $body['zip'],
  66. 'country' => $body['country'],
  67. 'kyc' => isset($body['kyc']) ? (int)$body['kyc'] : 0,
  68. 'birthdate' => (int)$body['birthdate'],
  69. 'cpf' => $body['cpf'],
  70. 'company_id' => $companyId,
  71. 'role_id' => $roleId
  72. ];
  73. $userData = $this->userModel->createUser($userPayload);
  74. if (!$userData) {
  75. $pdo->rollBack();
  76. return ResponseLib::sendFail("Email already exists or creation failed", [], "E_VALIDATE")->withStatus(400);
  77. }
  78. $bin = dirname(__DIR__) . '/bin/easycli';
  79. $result = BashExecutor::run($bin . ' polygon create-new-address');
  80. if ((int)($result['exitCode'] ?? 1) !== 0) {
  81. $cliOutput = trim((string)($result['output'] ?? ''));
  82. $cliError = trim((string)($result['error'] ?? ''));
  83. error_log('[CompanyWithUserController] easycli wallet generation failed. Output: ' . $cliOutput . ' | Error: ' . $cliError);
  84. $pdo->rollBack();
  85. return ResponseLib::sendFail("Wallet generation failed", ['error' => $result['error'] ?? ''], "E_INTERNAL")->withStatus(500);
  86. }
  87. $output = trim((string)($result['output'] ?? ''));
  88. $parsed = [];
  89. foreach (preg_split('/\r?\n/', $output) as $line) {
  90. $line = trim($line);
  91. if ($line === '' || strpos($line, '=') === false) { continue; }
  92. [$k, $v] = explode('=', $line, 2);
  93. $parsed[trim($k)] = trim($v);
  94. }
  95. if (!isset($parsed['privateKey'], $parsed['publicKey'], $parsed['address'])) {
  96. $pdo->rollBack();
  97. return ResponseLib::sendFail("Wallet parsing failed", ['raw' => $output], "E_INTERNAL")->withStatus(500);
  98. }
  99. $stmt = $pdo->prepare('SELECT chain_id FROM "chain" WHERE chain_name = :name');
  100. $stmt->execute(['name' => 'polygon']);
  101. $chainId = $stmt->fetchColumn();
  102. if (!$chainId) {
  103. $pdo->rollBack();
  104. return ResponseLib::sendFail("Chain not found", [], "E_DATABASE")->withStatus(500);
  105. }
  106. $stmt = $pdo->prepare('INSERT INTO "wallet" (company_id, wallet_public_key, wallet_address, wallet_private_key, wallet_flag, chain_id) VALUES (:company_id, :public_key, :address, :private_key, :flag, :chain_id) RETURNING wallet_id');
  107. $stmt->execute([
  108. 'company_id' => $companyId,
  109. 'public_key' => $parsed['publicKey'],
  110. 'address' => $parsed['address'],
  111. 'private_key' => $parsed['privateKey'],
  112. 'flag' => 'a',
  113. 'chain_id' => (int)$chainId
  114. ]);
  115. $walletId = (int)$stmt->fetchColumn();
  116. $pdo->commit();
  117. $analysisPayload = [
  118. 'corporate_name' => $body['company_user'],
  119. 'cnpj' => $body['cnpj'],
  120. ];
  121. error_log(sprintf(
  122. '[CompanyWithUserController] Sending company_user="%s" with analysis payload: %s',
  123. $body['company_user'],
  124. json_encode($analysisPayload, JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE)
  125. ));
  126. try {
  127. $tshield = $this->tshieldService->generateCompanyLink(
  128. (int)$userData['user_id'],
  129. $analysisPayload
  130. );
  131. } catch (\Throwable $e) {
  132. return ResponseLib::sendFail('TShield link generation failed: ' . $e->getMessage(), [], 'E_EXTERNAL')->withStatus(502);
  133. }
  134. $payload = [
  135. 'status' => 'ok',
  136. 'link' => $tshield['link'],
  137. 'numberToken' => $tshield['number'],
  138. 'data' => [
  139. 'link' => $tshield['link'],
  140. 'numberToken' => $tshield['number'],
  141. 'tshield' => $tshield,
  142. ]
  143. ];
  144. return Response::json($payload)->withStatus(201);
  145. } catch (\Throwable $e) {
  146. if (isset($pdo) && $pdo->inTransaction()) { $pdo->rollBack(); }
  147. return ResponseLib::sendFail($e->getMessage(), [], 'E_DATABASE')->withStatus(500);
  148. }
  149. }
  150. }