CompanyWithUserController.php 6.2 KB

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