<?php
namespace App\Service;
use App\Entity\SupplementType;
use App\Entity\Survey;
use App\Entity\SurveyRate;
use App\Entity\UserSurvey;
use Bexio\Client;
use Bexio\Resource\Accounts;
use Bexio\Resource\Contact;
use Bexio\Resource\Invoice;
use Bexio\Resource\Item;
use Doctrine\ORM\EntityManagerInterface;
use Throwable;
class BexioService
{
private $client = null;
/**
* @var EntityManagerInterface
*/
private $em;
private $envVerification;
public function __construct(EntityManagerInterface $em)
{
$this->em = $em;
$this->client = new Client();
$this->client->setAccessToken($_ENV['BEXIO_TOKEN']);
$this->envVerification = 'prod' == $_ENV['APP_ENV'];
// $this->envVerification = true;
}
public function getContactById($id)
{
$bexio = new Contact($this->client);
$results = $bexio->searchContacts([[
'field' => 'id',
'value' => $id,
'criteria' => '=',
]]);
if (count($results)) {
return $results[0];
}
return null;
}
public function getContact(UserSurvey $userSurvey)
{
$bexio = new Contact($this->client);
try {
$contact = $bexio->getContact($userSurvey->getUser()->getBexioContactId());
} catch (Throwable $exception) {
return null;
}
return $contact;
}
public function createContact(UserSurvey $userSurvey)
{
if ($this->envVerification) {
$bexio = new Contact($this->client);
$request_body = $this->prepareContactData($userSurvey);
$contact = $bexio->createContact($request_body);
$userSurvey->getUser()->setBexioContactId($contact->id);
$this->em->persist($userSurvey);
$this->em->flush();
return $contact;
}
return null;
}
private function prepareContactData(UserSurvey $userSurvey)
{
$user = $userSurvey->getUser();
$userFile = $userSurvey->getUserFile();
$address = '';
$phone = '';
$country = 8;
if ($userFile) {
if ('France' != $userFile->getResidence()) {
if ('Suisse' == $userFile->getResidence()) {
$country = 1;
}
}
$owner = $userSurvey->getUserFile()->getOwner();
if ($owner) {
$address = $owner->getAddress();
$phone = $owner->getPhoneNumber();
}
}
$request_body = [
'contact_type_id' => 2,
'name_1' => $user->getLastname(),
'name_2' => $user->getName(),
'address' => $address,
'country_id' => $country,
'mail' => $user->getEmail(),
'phone_fixed' => $phone,
'user_id' => 1,
'owner_id' => 1,
];
return $request_body;
}
public function editContact(UserSurvey $userSurvey)
{
$bexio = new Contact($this->client);
$request_body = $this->prepareContactData($userSurvey);
if ($this->envVerification) {
return $bexio->editContact($userSurvey->getUser()->getBexioContactId(), $request_body);
}
return null;
}
public function getProduct($id)
{
$item = new Item($this->client);
return $item->getItem($id);
}
public function getProducts(array $params = [])
{
$item = new Item($this->client);
return $item->getItems($params);
}
public function createInvoice(UserSurvey $userSurvey)
{
$bexioInvoice = new Invoice($this->client);
$order = $userSurvey->getOrder();
$items = [];
$account = 149;
if (!is_null($userSurvey->getSurvey()->getBexioAccountId())) {
$account = $userSurvey->getSurvey()->getBexioAccountId();
}
foreach ($userSurvey->getUserSurveyBexioArticles() as $userSurveyBexioArticle) {
$items[] = [
'type' => 'KbPositionArticle',
'amount' => $userSurveyBexioArticle->getQuantity(),
'account_id' => $account,
'tax_id' => 3,
'unit_price' => $userSurveyBexioArticle->getPrice(),
'article_id' => $userSurveyBexioArticle->getBexioArticleId(),
];
}
foreach ($order->getOrderSupplements() as $supplement) {
$items[] = [
'type' => 'KbPositionArticle',
'amount' => 1,
'account_id' => $account,
'tax_id' => 3,
'unit_price' => $supplement->calculateAmount($order->getAmountNoDiscount()),
'article_id' => $supplement->getSupplementType()->getBexioArticleId(),
];
}
$discounts = $order->getDiscounts();
foreach ($discounts as $discount) {
if ('AFFILIATION' == $discount->getType() and $discount->getUser() === $userSurvey->getUser()) {
$discountValue = $discount->getDiscount() * $order->getAffiliationCount();
} else {
$discountValue = $discount->getDiscount();
}
$invoiceDiscount = [
'type' => 'KbPositionDiscount',
'text' => 'Reduction (' . $discount->getCode() . ')',
'is_percentual' => '%' == $discount->getDiscountType(),
'value' => $discountValue,
];
if ($discount->getDiscount() > 0) {
$items[] = $invoiceDiscount;
}
}
$country = 'SUISSE';
if ('IF' == $userSurvey->getSurvey()->getSurveyCategory()->getCode()) {
$country = 'FRANCE';
}
$userSurveyName = $userSurvey->conditionalSurveyName();
if (strlen($userSurveyName) > 39) {
$userSurveyName = $userSurvey->getSurvey()->getShortName();
}
$title = $country . ' - ' . $userSurveyName . ' - année fiscale ' . strval(intval(date('Y')) - 1) . ' - ' . $userSurvey->getId();
if ('AS' == $userSurvey->getSurvey()->getSurveyCategory()->getCode()) {
$title = 'CMU(CNTFS) - ' . $userSurveyName . ' - ' . $userSurvey->getId();
}
$invoice = [
'title' => $title,
'contact_id' => $userSurvey->getUser()->getBexioContactId(),
'user_id' => 1,
'logopaper_id' => 1,
'language_id' => 2,
'bank_account_id' => 1,
'currency_id' => 1,
'payment_type_id' => 4,
'footer' => 'Nous sommes Ă votre disposition pour toutes questions.
Meilleures salutations',
'mwst_type' => 2,
'mwst_is_net' => true,
'show_position_taxes' => false,
'template_slug' => '669e591726dcf0fcd10e36f6',
'positions' => $items,
];
if ($this->envVerification) {
return $bexioInvoice->createInvoice($invoice);
}
return null;
}
public function issueInvoice($id)
{
if ($this->envVerification) {
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->issueInvoice($id);
}
return null;
}
public function createAdminInvoicePayment($id, $params)
{
if ($this->envVerification) {
$params['bank_account_id'] = 5;
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->createInvoicePayment($id, $params);
}
return null;
}
public function createInvoicePayment($id, $params)
{
if ($this->envVerification) {
$params['bank_account_id'] = 4;
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->createInvoicePayment($id, $params);
}
return null;
}
public function getPdf($id)
{
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->getPdf($id);
}
public function markInvoiceAsSent($id)
{
if ($this->envVerification) {
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->markInvoiceAsSent($id);
}
return null;
}
public function sendInvoice($id, $params)
{
if ($this->envVerification) {
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->sendInvoice($id, $params);
}
return null;
}
public function generateProducts($force = false)
{
$surveys = $this->em->getRepository(Survey::class)->findAll();
foreach ($surveys->getSurveyProducts() as $surveyProduct) {
foreach ($surveyProduct->getSurveyRates() as $surveyRate) {
if (is_null($surveyRate->getBexioArticleId()) || $force) {
$bexioId = $this->createBexioProduct($surveyRate);
$surveyRate->setBexioArticleId($bexioId);
$this->em->persist($surveyRate);
}
}
}
$this->em->flush();
}
public function createBexioProduct(SurveyRate $surveyRate, $newPrice = null)
{
$template = [
'user_id' => 1,
'article_type_id' => 2,
'contact_id' => null,
'deliverer_code' => null,
'deliverer_name' => null,
'deliverer_description' => null,
'intern_description' => '
<ul>
<li>%description%</li>
</ul>',
'sale_price' => null,
'purchase_total' => null,
'sale_total' => null,
'currency_id' => 1,
'tax_income_id' => 3,
'tax_expense_id' => 10,
'html_text' => null,
'article_group_id' => null,
];
$base = $template;
$prefix = 'W.';
if (true === $surveyRate->getSurveyProduct()->isIsMultiple()) {
$prefix .= 'S.';
}
$suffix = '';
if ('SANS_ENFANTS_-_CELIBATAIRE' == $surveyRate->getSurveyProduct()->getCode()) {
$suffix = '.1.0';
} elseif ('AVEC_ENFANT' == $surveyRate->getSurveyProduct()->getCode() || 'AVEC_ENFANT_CELIBATAIRE' == $surveyRate->getSurveyProduct()->getCode()) {
$suffix = '.1.1';
} elseif ('MARIE' == $surveyRate->getSurveyProduct()->getCode()) {
$suffix = '.2.8';
}
$base['intern_code'] = $prefix . $surveyRate->getSurveyProduct()->getSurvey()->getCode() . $suffix . '.' . mt_rand(1111, 9999);
$name = $surveyRate->getSurveyProduct()->getSurvey()->getName();
$description = str_replace('%description%', $surveyRate->getSurveyProduct()->getName(), $template['intern_description']);
if ('CMU' == $surveyRate->getSurveyProduct()->getSurvey()->getCode()) {
$name = 'Forfait rectification CNTFS';
$description = str_replace('%description%', 'Rectification de cotisations CNTFS', $template['intern_description']);
}
$base['intern_name'] = $name;
$base['intern_description'] = $description;
if (!is_null($newPrice)) {
$base['sale_price'] = $newPrice;
} else {
$base['sale_price'] = $surveyRate->getPrice();
}
if ($this->envVerification) {
$item = new Item($this->client);
$bexioArticle = $item->createItem($base);
return $bexioArticle->id;
}
return null;
}
public function cancelInvoice($id)
{
if ($this->envVerification) {
$bexioInvoice = new Invoice($this->client);
try {
return $bexioInvoice->cancelInvoice($id);
} catch (Throwable $exception) {
return null;
}
}
return null;
}
public function createBexioSupplement(SupplementType $supplementType)
{
$template = [
'user_id' => 1,
'article_type_id' => 2,
'contact_id' => null,
'deliverer_code' => null,
'deliverer_name' => null,
'deliverer_description' => null,
'intern_description' => '
<ul>
<li>%description%</li>
</ul>',
'sale_price' => null,
'purchase_total' => null,
'sale_total' => null,
'currency_id' => 1,
'tax_income_id' => 3,
'tax_expense_id' => 10,
'html_text' => null,
'article_group_id' => null,
];
$base = $template;
$base['intern_code'] = 'W.SUPP_' . mt_rand(1111, 9999);
$base['intern_name'] = $supplementType->__toString();
$base['intern_description'] = str_replace('%description%', $supplementType->__toString(), $template['intern_description']);
$base['sale_price'] = 0;
if ($this->envVerification) {
$item = new Item($this->client);
$bexioArticle = $item->createItem($base);
return $bexioArticle->id;
}
return null;
}
public function isPaid($id)
{
if (!$this->envVerification) {
return false;
}
$payments = $this->getInvoicePayments($id);
$invoice = $this->getInvoice($id);
$totalPaid = 0;
$invoiceTotal = 0;
if (is_object($invoice)) {
if (property_exists($invoice, 'total')) {
$invoiceTotal = $invoice->total;
}
}
if (is_array($payments)) {
foreach ($payments as $payment) {
if (property_exists($payment, 'value')) {
$totalPaid += $payment->value;
}
}
}
if ($totalPaid >= $invoiceTotal) {
return true;
}
return false;
}
public function getInvoicePayments($id)
{
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->getInvoicePayments($id);
}
public function getInvoice($id)
{
$bexioInvoice = new Invoice($this->client);
return $bexioInvoice->getInvoice($id);
}
}