You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

381 lines
18KB

  1. <?php
  2. /**
  3. * Copyright distrib (2018)
  4. *
  5. * contact@opendistrib.net
  6. *
  7. * Ce logiciel est un programme informatique servant à aider les producteurs
  8. * à distribuer leur production en circuits courts.
  9. *
  10. * Ce logiciel est régi par la licence CeCILL soumise au droit français et
  11. * respectant les principes de diffusion des logiciels libres. Vous pouvez
  12. * utiliser, modifier et/ou redistribuer ce programme sous les conditions
  13. * de la licence CeCILL telle que diffusée par le CEA, le CNRS et l'INRIA
  14. * sur le site "http://www.cecill.info".
  15. *
  16. * En contrepartie de l'accessibilité au code source et des droits de copie,
  17. * de modification et de redistribution accordés par cette licence, il n'est
  18. * offert aux utilisateurs qu'une garantie limitée. Pour les mêmes raisons,
  19. * seule une responsabilité restreinte pèse sur l'auteur du programme, le
  20. * titulaire des droits patrimoniaux et les concédants successifs.
  21. *
  22. * A cet égard l'attention de l'utilisateur est attirée sur les risques
  23. * associés au chargement, à l'utilisation, à la modification et/ou au
  24. * développement et à la reproduction du logiciel par l'utilisateur étant
  25. * donné sa spécificité de logiciel libre, qui peut le rendre complexe à
  26. * manipuler et qui le réserve donc à des développeurs et des professionnels
  27. * avertis possédant des connaissances informatiques approfondies. Les
  28. * utilisateurs sont donc invités à charger et tester l'adéquation du
  29. * logiciel à leurs besoins dans des conditions permettant d'assurer la
  30. * sécurité de leurs systèmes et ou de leurs données et, plus généralement,
  31. * à l'utiliser et l'exploiter dans les mêmes conditions de sécurité.
  32. *
  33. * Le fait que vous puissiez accéder à cet en-tête signifie que vous avez
  34. * pris connaissance de la licence CeCILL, et que vous en avez accepté les
  35. * termes.
  36. */
  37. namespace producer\controllers;
  38. use common\helpers\GlobalParam;
  39. use common\helpers\Mailjet;
  40. use common\helpers\MeanPayment;
  41. use common\logic\User\CreditHistory\CreditHistory;
  42. use common\logic\User\CreditHistory\CreditHistorySearch;
  43. use common\logic\User\User\User;
  44. use common\logic\User\UserProducer\UserProducer;
  45. use producer\models\CreditForm;
  46. use yii\filters\VerbFilter;
  47. class CreditController extends ProducerBaseController
  48. {
  49. /**
  50. * @inheritdoc
  51. */
  52. public function behaviors()
  53. {
  54. return [
  55. 'verbs' => [
  56. 'class' => VerbFilter::className(),
  57. 'actions' => [
  58. 'stripe-verification' => ['post'],
  59. ],
  60. ]
  61. ];
  62. }
  63. public function actions()
  64. {
  65. return [
  66. 'captcha' => [
  67. 'class' => 'yii\captcha\CaptchaAction',
  68. 'fixedVerifyCode' => YII_ENV_TEST ? 'testme' : null,
  69. ],
  70. ];
  71. }
  72. public function beforeAction($action)
  73. {
  74. $this->enableCsrfValidation = false;
  75. return parent::beforeAction($action);
  76. }
  77. /**
  78. * Affiche l'historique du crédit client.
  79. *
  80. */
  81. public function actionHistory($returnPayment = '')
  82. {
  83. $producer = $this->getProducer();
  84. if(\Yii::$app->user->isGuest) {
  85. return $this->redirect(Yii::$app->urlManagerFrontend->createAbsoluteUrl(['site/producer','id' => $producer->id])) ;
  86. }
  87. $searchModel = new CreditHistorySearch();
  88. $searchModel->id_user = GlobalParam::getCurrentUserId();
  89. $dataProvider = $searchModel->search(\Yii::$app->request->queryParams);
  90. $userProducer = UserProducer::searchOne([
  91. 'id_producer' => $producer->id,
  92. 'id_user' => GlobalParam::getCurrentUserId()
  93. ]);
  94. if (strlen($returnPayment)) {
  95. if ($returnPayment == 'success') {
  96. \Yii::$app->getSession()->setFlash('success', "Paiement accepté : votre compte vient d'être crédité.");
  97. }
  98. if ($returnPayment == 'cancel') {
  99. \Yii::$app->getSession()->setFlash('error', 'Paiement annulé.');
  100. }
  101. }
  102. return $this->render('history', [
  103. 'searchModel' => $searchModel,
  104. 'dataProvider' => $dataProvider,
  105. 'creditUser' => $userProducer->credit
  106. ]);
  107. }
  108. public function actionAdd()
  109. {
  110. $producer = $this->getProducer();
  111. if(\Yii::$app->user->isGuest) {
  112. return $this->redirect(\Yii::$app->urlManagerFrontend->createAbsoluteUrl(['site/producer','id' => $producer->id])) ;
  113. }
  114. if ($producer->online_payment || $producer->option_stripe_mode_test) {
  115. $creditForm = new CreditForm;
  116. if ($creditForm->load(\Yii::$app->request->post()) && $creditForm->validate()) {
  117. $user = User::getCurrent();
  118. \Stripe\Stripe::setApiKey(
  119. $producer->getPrivateKeyApiStripe()
  120. );
  121. $checkout_session = \Stripe\Checkout\Session::create([
  122. 'line_items' => [
  123. [
  124. 'price_data' => [
  125. 'currency' => 'eur',
  126. 'product_data' => [
  127. 'name' => 'Alimentation crédit',
  128. ],
  129. 'unit_amount' => (float)$creditForm->amount * 100,
  130. ],
  131. 'quantity' => 1,
  132. ]
  133. ],
  134. 'payment_method_types' => ['card'],
  135. 'mode' => 'payment',
  136. 'customer_email' => $user->email,
  137. 'client_reference_id' => $user->id,
  138. 'payment_intent_data' => [
  139. 'metadata' => [
  140. 'user_id' => $user->id,
  141. 'producer_id' => $producer->id
  142. ],
  143. ],
  144. 'success_url' => \Yii::$app->urlManagerProducer->createAbsoluteUrl(
  145. [
  146. 'credit/history',
  147. 'returnPayment' => 'success'
  148. ]
  149. ),
  150. 'cancel_url' => \Yii::$app->urlManagerProducer->createAbsoluteUrl(
  151. [
  152. 'credit/history',
  153. 'returnPayment' => 'cancel'
  154. ]
  155. ),
  156. ]);
  157. header("HTTP/1.1 303 See Other");
  158. header("Location: " . $checkout_session->url);
  159. die();
  160. }
  161. return $this->render('add', [
  162. 'creditForm' => $creditForm
  163. ]);
  164. } else {
  165. throw new \yii\base\UserException('Cette option est désactivée chez ce producteur.');
  166. }
  167. }
  168. public function actionStripeVerification()
  169. {
  170. $creditHistoryBuilder = $this->getLogic()->getCreditHistoryContainer()->getBuilder();
  171. $producer = $this->getProducer();
  172. $contactProducer = $producer->getMainContact();
  173. $payload = @file_get_contents('php://input');
  174. $sig_header = $_SERVER['HTTP_STRIPE_SIGNATURE'];
  175. $event = null;
  176. try {
  177. $event = \Stripe\Webhook::constructEvent(
  178. $payload,
  179. $sig_header,
  180. $producer->getPrivateKeyEndpointStripe()
  181. );
  182. } catch (\UnexpectedValueException $e) {
  183. // Invalid payload
  184. http_response_code(400);
  185. exit();
  186. } catch (\Stripe\Exception\SignatureVerificationException $e) {
  187. // Invalid signature
  188. http_response_code(400);
  189. exit();
  190. }
  191. $paymentIntent = $event->data->object;
  192. $paymentIntentMetadata = $paymentIntent->metadata;
  193. $amount = $paymentIntent->amount / 100;
  194. $idUser = $paymentIntentMetadata->user_id;
  195. $user = User::findOne($paymentIntentMetadata->user_id);
  196. $idProducer = $paymentIntentMetadata->producer_id;
  197. if(isset($paymentIntentMetadata->order_id)) {
  198. $order = Order::searchOne([
  199. 'id' => $paymentIntentMetadata->order_id
  200. ]);
  201. $pointSale = PointSale::searchOne(['id' => $order->id_point_sale]);
  202. $distribution = DistributionModel::find()->where(['id' => $order->id_distribution])->one();
  203. }
  204. // Handle the event
  205. switch ($event->type) {
  206. case 'charge.succeeded':
  207. $creditHistoryExist = CreditHistory::searchOne([
  208. 'id_user' => $idUser,
  209. 'amount' => $amount,
  210. ], [
  211. 'conditions' => [
  212. 'date > DATE_SUB(NOW(), INTERVAL 1 MINUTE)'
  213. ]
  214. ]);
  215. if(!$creditHistoryExist) {
  216. // on crédite le crédit du client
  217. $creditHistory = new CreditHistory();
  218. $creditHistory->id_user = $idUser;
  219. $creditHistory->id_user_action = $idUser;
  220. $creditHistory->id_producer = $idProducer;
  221. $creditHistory->type = CreditHistory::TYPE_CREDIT;
  222. $creditHistory->comment = null;
  223. $creditHistory->amount = $amount;
  224. $creditHistory->mean_payment = MeanPayment::CREDIT_CARD;
  225. $creditHistoryBuilder->save($creditHistory);
  226. if (isset($order) && $order) {
  227. // paiement de la commande
  228. $order->saveCreditHistory(
  229. CreditHistory::TYPE_PAYMENT,
  230. $amount,
  231. $idProducer,
  232. $order->id_user,
  233. $order->id_user
  234. );
  235. // client : envoi d'un email de confirmation de paiement
  236. $paramsEmail = [
  237. 'from_email' => $producer->getEmailOpendistrib(),
  238. 'from_name' => $producer->name,
  239. 'to_email' => $user->email,
  240. 'to_name' => $user->getUsername(),
  241. 'subject' => '[' . $producer->name . '] Confirmation de paiement',
  242. 'content_view_text' => '@common/mail/paymentOrderConfirm-text.php',
  243. 'content_view_html' => '@common/mail/paymentOrderConfirm-html.php',
  244. 'content_params' => [
  245. 'amount' => $amount,
  246. 'user' => $user,
  247. 'producer' => $producer,
  248. ]
  249. ];
  250. //Mailjet::sendMail($paramsEmail);
  251. // producteur : mail de confirmation
  252. Mailjet::sendMail([
  253. 'from_email' => $producer->getEmailOpendistrib(),
  254. 'from_name' => $producer->name,
  255. 'to_email' => $contactProducer->email,
  256. 'to_name' => $contactProducer->name,
  257. 'subject' => '[' . $producer->name . '] Confirmation de commande',
  258. 'content_view_text' => '@common/mail/orderConfirmProducer-text.php',
  259. 'content_view_html' => '@common/mail/orderConfirmProducer-html.php',
  260. 'content_params' => [
  261. 'order' => $order,
  262. 'pointSale' => $pointSale,
  263. 'distribution' => $distribution,
  264. 'user' => $user,
  265. 'producer' => $producer
  266. ]
  267. ]);
  268. } else {
  269. // envoi d'un email de confirmation
  270. $userProducer = UserProducer::find()
  271. ->where([
  272. 'id_user' => $idUser,
  273. 'id_producer' => $idProducer
  274. ])
  275. ->one();
  276. Mailjet::sendMail([
  277. 'from_email' => $producer->getEmailOpendistrib(),
  278. 'from_name' => $producer->name,
  279. 'to_email' => $user->email,
  280. 'to_name' => $user->getUsername(),
  281. 'subject' => '[' . $producer->name . '] Alimentation de votre crédit',
  282. 'content_view_text' => '@common/mail/creditConfirm-text.php',
  283. 'content_view_html' => '@common/mail/creditConfirm-html.php',
  284. 'content_params' => [
  285. 'user' => $user,
  286. 'userProducer' => $userProducer,
  287. 'producer' => $producer,
  288. 'amount' => $amount,
  289. ]
  290. ]);
  291. }
  292. }
  293. break;
  294. case 'charge.failed':
  295. // client
  296. Mailjet::sendMail([
  297. 'from_email' => $producer->getEmailOpendistrib(),
  298. 'from_name' => $producer->name,
  299. 'to_email' => $user->email,
  300. 'to_name' => $user->getUsername(),
  301. 'subject' => '['.$producer->name.'] Erreur de paiement',
  302. 'content_view_text' => '@common/mail/paymentError-text.php',
  303. 'content_view_html' => '@common/mail/paymentError-html.php',
  304. 'content_params' => [
  305. 'amount' => $amount,
  306. 'user' => $user,
  307. 'producer' => $producer,
  308. ]
  309. ]);
  310. // producteur
  311. if(isset($order) && $order) {
  312. Mailjet::sendMail([
  313. 'from_email' => $producer->getEmailOpendistrib(),
  314. 'from_name' => $producer->name,
  315. 'to_email' => $contactProducer->email,
  316. 'to_name' => $contactProducer->name,
  317. 'subject' => '[' . $producer->name . '] Erreur de paiement',
  318. 'content_view_text' => '@common/mail/paymentErrorProducer-text.php',
  319. 'content_view_html' => '@common/mail/paymentErrorProducer-html.php',
  320. 'content_params' => [
  321. 'amount' => $amount,
  322. 'user' => $user,
  323. 'producer' => $producer,
  324. 'order' => $order,
  325. 'distribution' => $distribution
  326. ]
  327. ]);
  328. }
  329. break;
  330. // handle other event types
  331. default:
  332. echo 'Received unknown event type ' . $event->type;
  333. }
  334. http_response_code(200);
  335. die();
  336. }
  337. }