CompanyWithUserController.php 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  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, 50))
  36. ->assert($body);
  37. } catch (ValidationException $e) {
  38. return ResponseLib::sendFail("Validation failed: " . $e->getFullMessage(), [], "E_VALIDATE")->withStatus(400);
  39. }
  40. try {
  41. $pdo = $GLOBALS['pdo'];
  42. $pdo->beginTransaction();
  43. $companyId = $this->companyModel->createCompany($body['company_name'], 'a');
  44. $roleId = 1;
  45. $chk = $pdo->prepare('SELECT 1 FROM "role" WHERE role_id = :rid');
  46. $chk->execute(['rid' => $roleId]);
  47. if (!$chk->fetchColumn()) {
  48. $pdo->rollBack();
  49. return ResponseLib::sendFail('Default role_id 1 not found', [], 'E_DATABASE')->withStatus(500);
  50. }
  51. $userPayload = [
  52. 'username' => $body['username'],
  53. 'email' => $body['email'],
  54. 'password' => $body['password'],
  55. 'phone' => $body['phone'],
  56. 'address' => $body['address'],
  57. 'city' => $body['city'],
  58. 'state' => $body['state'],
  59. 'zip' => $body['zip'],
  60. 'country' => $body['country'],
  61. 'kyc' => (int)$body['kyc'],
  62. 'birthdate' => (int)$body['birthdate'],
  63. 'cpf' => $body['cpf'],
  64. 'company_id' => $companyId,
  65. 'role_id' => $roleId
  66. ];
  67. $userData = $this->userModel->createUser($userPayload);
  68. if (!$userData) {
  69. $pdo->rollBack();
  70. return ResponseLib::sendFail("Email already exists or creation failed", [], "E_VALIDATE")->withStatus(400);
  71. }
  72. $bin = dirname(__DIR__) . '/bin/easycli';
  73. $result = BashExecutor::run($bin . ' polygon create-new-address');
  74. if (($result['exitCode'] ?? 1) !== 0) {
  75. $pdo->rollBack();
  76. return ResponseLib::sendFail("Wallet generation failed", ['error' => $result['error'] ?? ''], "E_INTERNAL")->withStatus(500);
  77. }
  78. $output = trim((string)($result['output'] ?? ''));
  79. $parsed = [];
  80. foreach (preg_split('/\r?\n/', $output) as $line) {
  81. $line = trim($line);
  82. if ($line === '' || strpos($line, '=') === false) { continue; }
  83. [$k, $v] = explode('=', $line, 2);
  84. $parsed[trim($k)] = trim($v);
  85. }
  86. if (!isset($parsed['privateKey'], $parsed['publicKey'], $parsed['address'])) {
  87. $pdo->rollBack();
  88. return ResponseLib::sendFail("Wallet parsing failed", ['raw' => $output], "E_INTERNAL")->withStatus(500);
  89. }
  90. $stmt = $pdo->prepare('SELECT chain_id FROM "chain" WHERE chain_name = :name');
  91. $stmt->execute(['name' => 'polygon']);
  92. $chainId = $stmt->fetchColumn();
  93. if (!$chainId) {
  94. $pdo->rollBack();
  95. return ResponseLib::sendFail("Chain not found", [], "E_DATABASE")->withStatus(500);
  96. }
  97. $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');
  98. $stmt->execute([
  99. 'company_id' => $companyId,
  100. 'public_key' => $parsed['publicKey'],
  101. 'address' => $parsed['address'],
  102. 'private_key' => $parsed['privateKey'],
  103. 'flag' => 'a',
  104. 'chain_id' => (int)$chainId
  105. ]);
  106. $walletId = (int)$stmt->fetchColumn();
  107. $pdo->commit();
  108. return ResponseLib::sendOk([
  109. 'company_id' => $companyId,
  110. 'role_id' => $roleId,
  111. 'user' => $userData,
  112. 'wallet_id' => $walletId,
  113. 'wallet_address' => $parsed['address']
  114. ], 'S_CREATED');
  115. } catch (\Throwable $e) {
  116. if (isset($pdo) && $pdo->inTransaction()) { $pdo->rollBack(); }
  117. return ResponseLib::sendFail($e->getMessage(), [], 'E_DATABASE')->withStatus(500);
  118. }
  119. }
  120. }